From dea17d475bb1dca85e0c6c249c06d95abaef2239 Mon Sep 17 00:00:00 2001 From: david-leifker <114954101+david-leifker@users.noreply.github.com> Date: Mon, 30 Sep 2024 14:28:30 -0500 Subject: [PATCH 1/9] chore(cleanup): remove legacy bootstrap step (#11494) --- .../boot/MCLBootstrapManagerFactory.java | 4 - .../boot/MCPBootstrapManagerFactory.java | 4 - .../src/main/resources/application.yaml | 6 - .../factories/BootstrapManagerFactory.java | 16 - .../boot/steps/BackfillBrowsePathsV2Step.java | 156 ------- .../steps/UpgradeDefaultBrowsePathsStep.java | 145 ------- .../steps/BackfillBrowsePathsV2StepTest.java | 188 -------- .../boot/steps/IngestEntityTypesStepTest.java | 8 + .../UpgradeDefaultBrowsePathsStepTest.java | 406 ------------------ 9 files changed, 8 insertions(+), 925 deletions(-) delete mode 100644 metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2Step.java delete mode 100644 metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStep.java delete mode 100644 metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2StepTest.java delete mode 100644 metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStepTest.java diff --git a/metadata-jobs/mae-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCLBootstrapManagerFactory.java b/metadata-jobs/mae-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCLBootstrapManagerFactory.java index 8ad1638115dae..8a913910b3523 100644 --- a/metadata-jobs/mae-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCLBootstrapManagerFactory.java +++ b/metadata-jobs/mae-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCLBootstrapManagerFactory.java @@ -11,7 +11,6 @@ import javax.annotation.Nonnull; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; -import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; @@ -27,9 +26,6 @@ public class MCLBootstrapManagerFactory { @Autowired private ConfigurationProvider _configurationProvider; - @Value("${bootstrap.upgradeDefaultBrowsePaths.enabled}") - private Boolean _upgradeDefaultBrowsePathsEnabled; - @Bean(name = "mclBootstrapManager") @Scope("singleton") @Nonnull diff --git a/metadata-jobs/mce-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCPBootstrapManagerFactory.java b/metadata-jobs/mce-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCPBootstrapManagerFactory.java index 0220764cd99d6..5419fa48ee5ba 100644 --- a/metadata-jobs/mce-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCPBootstrapManagerFactory.java +++ b/metadata-jobs/mce-consumer/src/main/java/com/linkedin/metadata/kafka/boot/MCPBootstrapManagerFactory.java @@ -11,7 +11,6 @@ import javax.annotation.Nonnull; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Qualifier; -import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Conditional; import org.springframework.context.annotation.Configuration; @@ -27,9 +26,6 @@ public class MCPBootstrapManagerFactory { @Autowired private ConfigurationProvider _configurationProvider; - @Value("${bootstrap.upgradeDefaultBrowsePaths.enabled}") - private Boolean _upgradeDefaultBrowsePathsEnabled; - @Bean(name = "mcpBootstrapManager") @Scope("singleton") @Nonnull diff --git a/metadata-service/configuration/src/main/resources/application.yaml b/metadata-service/configuration/src/main/resources/application.yaml index 0ce0b976c976e..5e07bfc479e93 100644 --- a/metadata-service/configuration/src/main/resources/application.yaml +++ b/metadata-service/configuration/src/main/resources/application.yaml @@ -343,12 +343,6 @@ incidents: consumerGroupSuffix: ${INCIDENTS_HOOK_CONSUMER_GROUP_SUFFIX:} bootstrap: - upgradeDefaultBrowsePaths: - enabled: ${UPGRADE_DEFAULT_BROWSE_PATHS_ENABLED:false} # enable to run the upgrade to migrate legacy default browse paths to new ones - backfillBrowsePathsV2: - enabled: ${BACKFILL_BROWSE_PATHS_V2:false} # Enables running the backfill of browsePathsV2 upgrade step. There are concerns about the load of this step so hiding it behind a flag. Deprecating in favor of running through SystemUpdate - reprocessDefaultBrowsePathsV2: - enabled: ${REPROCESS_DEFAULT_BROWSE_PATHS_V2:false} # reprocess V2 browse paths which were set to the default: {"path":[{"id":"Default"}]} policies: file: ${BOOTSTRAP_POLICIES_FILE:classpath:boot/policies.json} # eg for local file diff --git a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/factories/BootstrapManagerFactory.java b/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/factories/BootstrapManagerFactory.java index 97a009dcbbb6d..9e29883f439a7 100644 --- a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/factories/BootstrapManagerFactory.java +++ b/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/factories/BootstrapManagerFactory.java @@ -9,7 +9,6 @@ import com.linkedin.metadata.boot.BootstrapManager; import com.linkedin.metadata.boot.BootstrapStep; import com.linkedin.metadata.boot.dependencies.BootstrapDependency; -import com.linkedin.metadata.boot.steps.BackfillBrowsePathsV2Step; import com.linkedin.metadata.boot.steps.IndexDataPlatformsStep; import com.linkedin.metadata.boot.steps.IngestDataPlatformInstancesStep; import com.linkedin.metadata.boot.steps.IngestDataPlatformsStep; @@ -25,7 +24,6 @@ import com.linkedin.metadata.boot.steps.RestoreColumnLineageIndices; import com.linkedin.metadata.boot.steps.RestoreDbtSiblingsIndices; import com.linkedin.metadata.boot.steps.RestoreGlossaryIndices; -import com.linkedin.metadata.boot.steps.UpgradeDefaultBrowsePathsStep; import com.linkedin.metadata.boot.steps.WaitForSystemUpdateStep; import com.linkedin.metadata.entity.AspectMigrationsDao; import com.linkedin.metadata.entity.EntityService; @@ -89,12 +87,6 @@ public class BootstrapManagerFactory { @Autowired private ConfigurationProvider _configurationProvider; - @Value("${bootstrap.upgradeDefaultBrowsePaths.enabled}") - private Boolean _upgradeDefaultBrowsePathsEnabled; - - @Value("${bootstrap.backfillBrowsePathsV2.enabled}") - private Boolean _backfillBrowsePathsV2Enabled; - @Value("${bootstrap.policies.file}") private Resource _policiesResource; @@ -154,14 +146,6 @@ protected BootstrapManager createInstance( ingestDataTypesStep, ingestEntityTypesStep)); - if (_upgradeDefaultBrowsePathsEnabled) { - finalSteps.add(new UpgradeDefaultBrowsePathsStep(_entityService)); - } - - if (_backfillBrowsePathsV2Enabled) { - finalSteps.add(new BackfillBrowsePathsV2Step(_entityService, _searchService)); - } - return new BootstrapManager(finalSteps); } } diff --git a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2Step.java b/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2Step.java deleted file mode 100644 index 2c00c73c96549..0000000000000 --- a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2Step.java +++ /dev/null @@ -1,156 +0,0 @@ -package com.linkedin.metadata.boot.steps; - -import static com.linkedin.metadata.utils.CriterionUtils.buildExistsCriterion; -import static com.linkedin.metadata.utils.CriterionUtils.buildIsNullCriterion; -import static com.linkedin.metadata.utils.SystemMetadataUtils.createDefaultSystemMetadata; - -import com.google.common.collect.ImmutableList; -import com.google.common.collect.ImmutableSet; -import com.linkedin.common.AuditStamp; -import com.linkedin.common.BrowsePathsV2; -import com.linkedin.common.urn.Urn; -import com.linkedin.events.metadata.ChangeType; -import com.linkedin.metadata.Constants; -import com.linkedin.metadata.aspect.utils.DefaultAspectsUtil; -import com.linkedin.metadata.boot.UpgradeStep; -import com.linkedin.metadata.entity.EntityService; -import com.linkedin.metadata.query.filter.ConjunctiveCriterion; -import com.linkedin.metadata.query.filter.ConjunctiveCriterionArray; -import com.linkedin.metadata.query.filter.Criterion; -import com.linkedin.metadata.query.filter.CriterionArray; -import com.linkedin.metadata.query.filter.Filter; -import com.linkedin.metadata.search.ScrollResult; -import com.linkedin.metadata.search.SearchEntity; -import com.linkedin.metadata.search.SearchService; -import com.linkedin.metadata.utils.GenericRecordUtils; -import com.linkedin.mxe.MetadataChangeProposal; -import io.datahubproject.metadata.context.OperationContext; -import java.util.Set; -import javax.annotation.Nonnull; -import lombok.extern.slf4j.Slf4j; - -@Slf4j -public class BackfillBrowsePathsV2Step extends UpgradeStep { - - private static final Set ENTITY_TYPES_TO_MIGRATE = - ImmutableSet.of( - Constants.DATASET_ENTITY_NAME, - Constants.DASHBOARD_ENTITY_NAME, - Constants.CHART_ENTITY_NAME, - Constants.DATA_JOB_ENTITY_NAME, - Constants.DATA_FLOW_ENTITY_NAME, - Constants.ML_MODEL_ENTITY_NAME, - Constants.ML_MODEL_GROUP_ENTITY_NAME, - Constants.ML_FEATURE_TABLE_ENTITY_NAME, - Constants.ML_FEATURE_ENTITY_NAME); - private static final String VERSION = "2"; - private static final String UPGRADE_ID = "backfill-default-browse-paths-v2-step"; - private static final Integer BATCH_SIZE = 5000; - - private final SearchService searchService; - - public BackfillBrowsePathsV2Step(EntityService entityService, SearchService searchService) { - super(entityService, VERSION, UPGRADE_ID); - this.searchService = searchService; - } - - @Nonnull - @Override - public ExecutionMode getExecutionMode() { - return ExecutionMode.BLOCKING; // ensure there are no write conflicts. - } - - @Override - public void upgrade(@Nonnull OperationContext systemOpContext) throws Exception { - final AuditStamp auditStamp = - new AuditStamp() - .setActor(Urn.createFromString(Constants.SYSTEM_ACTOR)) - .setTime(System.currentTimeMillis()); - - String scrollId = null; - for (String entityType : ENTITY_TYPES_TO_MIGRATE) { - int migratedCount = 0; - do { - log.info( - String.format( - "Upgrading batch %s-%s of browse paths for entity type %s", - migratedCount, migratedCount + BATCH_SIZE, entityType)); - scrollId = backfillBrowsePathsV2(systemOpContext, entityType, auditStamp, scrollId); - migratedCount += BATCH_SIZE; - } while (scrollId != null); - } - } - - private String backfillBrowsePathsV2( - @Nonnull OperationContext systemOperationContext, - String entityType, - AuditStamp auditStamp, - String scrollId) - throws Exception { - - // Condition: has `browsePaths` AND does NOT have `browsePathV2` - Criterion missingBrowsePathV2 = buildIsNullCriterion("browsePathV2"); - - // Excludes entities without browsePaths - Criterion hasBrowsePathV1 = buildExistsCriterion("browsePaths"); - - CriterionArray criterionArray = new CriterionArray(); - criterionArray.add(missingBrowsePathV2); - criterionArray.add(hasBrowsePathV1); - - ConjunctiveCriterion conjunctiveCriterion = new ConjunctiveCriterion(); - conjunctiveCriterion.setAnd(criterionArray); - - ConjunctiveCriterionArray conjunctiveCriterionArray = new ConjunctiveCriterionArray(); - conjunctiveCriterionArray.add(conjunctiveCriterion); - - Filter filter = new Filter(); - filter.setOr(conjunctiveCriterionArray); - - final ScrollResult scrollResult = - searchService.scrollAcrossEntities( - systemOperationContext, - ImmutableList.of(entityType), - "*", - filter, - null, - scrollId, - "5m", - BATCH_SIZE); - if (scrollResult.getNumEntities() == 0 || scrollResult.getEntities().size() == 0) { - return null; - } - - for (SearchEntity searchEntity : scrollResult.getEntities()) { - try { - ingestBrowsePathsV2(systemOperationContext, searchEntity.getEntity(), auditStamp); - } catch (Exception e) { - // don't stop the whole step because of one bad urn or one bad ingestion - log.error( - String.format( - "Error ingesting default browsePathsV2 aspect for urn %s", - searchEntity.getEntity()), - e); - } - } - - return scrollResult.getScrollId(); - } - - private void ingestBrowsePathsV2( - @Nonnull OperationContext systemOperationContext, Urn urn, AuditStamp auditStamp) - throws Exception { - BrowsePathsV2 browsePathsV2 = - DefaultAspectsUtil.buildDefaultBrowsePathV2( - systemOperationContext, urn, true, entityService); - log.debug(String.format("Adding browse path v2 for urn %s with value %s", urn, browsePathsV2)); - MetadataChangeProposal proposal = new MetadataChangeProposal(); - proposal.setEntityUrn(urn); - proposal.setEntityType(urn.getEntityType()); - proposal.setAspectName(Constants.BROWSE_PATHS_V2_ASPECT_NAME); - proposal.setChangeType(ChangeType.UPSERT); - proposal.setSystemMetadata(createDefaultSystemMetadata()); - proposal.setAspect(GenericRecordUtils.serializeAspect(browsePathsV2)); - entityService.ingestProposal(systemOperationContext, proposal, auditStamp, false); - } -} diff --git a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStep.java b/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStep.java deleted file mode 100644 index 89846476a9875..0000000000000 --- a/metadata-service/factories/src/main/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStep.java +++ /dev/null @@ -1,145 +0,0 @@ -package com.linkedin.metadata.boot.steps; - -import static com.linkedin.metadata.Constants.*; -import static com.linkedin.metadata.utils.SystemMetadataUtils.createDefaultSystemMetadata; - -import com.google.common.collect.ImmutableSet; -import com.linkedin.common.AuditStamp; -import com.linkedin.common.BrowsePaths; -import com.linkedin.common.urn.Urn; -import com.linkedin.data.template.RecordTemplate; -import com.linkedin.events.metadata.ChangeType; -import com.linkedin.metadata.Constants; -import com.linkedin.metadata.aspect.utils.DefaultAspectsUtil; -import com.linkedin.metadata.boot.UpgradeStep; -import com.linkedin.metadata.entity.EntityService; -import com.linkedin.metadata.entity.ListResult; -import com.linkedin.metadata.query.ExtraInfo; -import com.linkedin.metadata.search.utils.BrowsePathUtils; -import com.linkedin.metadata.utils.GenericRecordUtils; -import com.linkedin.mxe.MetadataChangeProposal; -import io.datahubproject.metadata.context.OperationContext; -import java.util.Set; -import javax.annotation.Nonnull; -import lombok.extern.slf4j.Slf4j; - -/** - * This is an opt-in optional upgrade step to migrate your browse paths to the new truncated form. - * It is idempotent, can be retried as many times as necessary. - */ -@Slf4j -public class UpgradeDefaultBrowsePathsStep extends UpgradeStep { - - private static final Set ENTITY_TYPES_TO_MIGRATE = - ImmutableSet.of( - Constants.DATASET_ENTITY_NAME, - Constants.DASHBOARD_ENTITY_NAME, - Constants.CHART_ENTITY_NAME, - Constants.DATA_JOB_ENTITY_NAME, - Constants.DATA_FLOW_ENTITY_NAME); - private static final String VERSION = "1"; - private static final String UPGRADE_ID = "upgrade-default-browse-paths-step"; - private static final Integer BATCH_SIZE = 5000; - - public UpgradeDefaultBrowsePathsStep(EntityService entityService) { - super(entityService, VERSION, UPGRADE_ID); - } - - @Override - public void upgrade(@Nonnull OperationContext systemOperationContext) throws Exception { - final AuditStamp auditStamp = - new AuditStamp() - .setActor(Urn.createFromString(Constants.SYSTEM_ACTOR)) - .setTime(System.currentTimeMillis()); - - int total = 0; - for (String entityType : ENTITY_TYPES_TO_MIGRATE) { - int migratedCount = 0; - do { - log.info( - String.format( - "Upgrading batch %s-%s out of %s of browse paths for entity type %s", - migratedCount, migratedCount + BATCH_SIZE, total, entityType)); - total = - getAndMigrateBrowsePaths(systemOperationContext, entityType, migratedCount, auditStamp); - migratedCount += BATCH_SIZE; - } while (migratedCount < total); - } - log.info("Successfully upgraded all browse paths!"); - } - - @Nonnull - @Override - public ExecutionMode getExecutionMode() { - return ExecutionMode.BLOCKING; // ensure there are no write conflicts. - } - - private int getAndMigrateBrowsePaths( - @Nonnull OperationContext opContext, String entityType, int start, AuditStamp auditStamp) - throws Exception { - - final ListResult latestAspects = - entityService.listLatestAspects( - opContext, entityType, Constants.BROWSE_PATHS_ASPECT_NAME, start, BATCH_SIZE); - - if (latestAspects.getTotalCount() == 0 - || latestAspects.getValues() == null - || latestAspects.getMetadata() == null) { - log.debug( - String.format( - "Found 0 browse paths for entity with type %s. Skipping migration!", entityType)); - return 0; - } - - if (latestAspects.getValues().size() != latestAspects.getMetadata().getExtraInfos().size()) { - // Bad result -- we should log that we cannot migrate this batch of paths. - log.warn( - "Failed to match browse path aspects with corresponding urns. Found mismatched length between aspects ({})" - + "and metadata ({}) for metadata {}", - latestAspects.getValues().size(), - latestAspects.getMetadata().getExtraInfos().size(), - latestAspects.getMetadata()); - return latestAspects.getTotalCount(); - } - - for (int i = 0; i < latestAspects.getValues().size(); i++) { - - ExtraInfo info = latestAspects.getMetadata().getExtraInfos().get(i); - RecordTemplate browsePathsRec = latestAspects.getValues().get(i); - - // Assert on 2 conditions: - // 1. The latest browse path aspect contains only 1 browse path - // 2. The latest browse path matches exactly the legacy default path - - Urn urn = info.getUrn(); - BrowsePaths browsePaths = (BrowsePaths) browsePathsRec; - - log.debug(String.format("Inspecting browse path for urn %s, value %s", urn, browsePaths)); - - if (browsePaths.hasPaths() && browsePaths.getPaths().size() == 1) { - String legacyBrowsePath = - BrowsePathUtils.getLegacyDefaultBrowsePath(urn, opContext.getEntityRegistry()); - log.debug(String.format("Legacy browse path for urn %s, value %s", urn, legacyBrowsePath)); - if (legacyBrowsePath.equals(browsePaths.getPaths().get(0))) { - migrateBrowsePath(opContext, urn, auditStamp); - } - } - } - - return latestAspects.getTotalCount(); - } - - private void migrateBrowsePath( - @Nonnull OperationContext opContext, Urn urn, AuditStamp auditStamp) throws Exception { - BrowsePaths newPaths = DefaultAspectsUtil.buildDefaultBrowsePath(opContext, urn, entityService); - log.debug(String.format("Updating browse path for urn %s to value %s", urn, newPaths)); - MetadataChangeProposal proposal = new MetadataChangeProposal(); - proposal.setEntityUrn(urn); - proposal.setEntityType(urn.getEntityType()); - proposal.setAspectName(Constants.BROWSE_PATHS_ASPECT_NAME); - proposal.setChangeType(ChangeType.UPSERT); - proposal.setSystemMetadata(createDefaultSystemMetadata()); - proposal.setAspect(GenericRecordUtils.serializeAspect(newPaths)); - entityService.ingestProposal(opContext, proposal, auditStamp, false); - } -} diff --git a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2StepTest.java b/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2StepTest.java deleted file mode 100644 index 54c2cd73dfe1d..0000000000000 --- a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/BackfillBrowsePathsV2StepTest.java +++ /dev/null @@ -1,188 +0,0 @@ -package com.linkedin.metadata.boot.steps; - -import static com.linkedin.metadata.Constants.CONTAINER_ASPECT_NAME; -import static org.mockito.ArgumentMatchers.any; -import static org.mockito.Mockito.mock; - -import com.google.common.collect.ImmutableList; -import com.linkedin.common.AuditStamp; -import com.linkedin.common.urn.Urn; -import com.linkedin.common.urn.UrnUtils; -import com.linkedin.entity.Aspect; -import com.linkedin.entity.EntityResponse; -import com.linkedin.entity.EnvelopedAspect; -import com.linkedin.entity.EnvelopedAspectMap; -import com.linkedin.metadata.Constants; -import com.linkedin.metadata.entity.EntityService; -import com.linkedin.metadata.query.filter.Filter; -import com.linkedin.metadata.search.ScrollResult; -import com.linkedin.metadata.search.SearchEntity; -import com.linkedin.metadata.search.SearchEntityArray; -import com.linkedin.metadata.search.SearchService; -import com.linkedin.mxe.MetadataChangeProposal; -import io.datahubproject.metadata.context.OperationContext; -import io.datahubproject.test.metadata.context.TestOperationContexts; -import java.net.URISyntaxException; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import org.mockito.Mockito; -import org.testng.annotations.Test; - -public class BackfillBrowsePathsV2StepTest { - private static final String VERSION = "2"; - private static final String UPGRADE_URN = - String.format( - "urn:li:%s:%s", - Constants.DATA_HUB_UPGRADE_ENTITY_NAME, "backfill-default-browse-paths-v2-step"); - - private static final String DATASET_URN = - "urn:li:dataset:(urn:li:dataPlatform:platform,name,PROD)"; - private static final String DASHBOARD_URN = "urn:li:dashboard:(airflow,id)"; - private static final String CHART_URN = "urn:li:chart:(looker,baz)"; - private static final String DATA_JOB_URN = - "urn:li:dataJob:(urn:li:dataFlow:(airflow,test,prod),test1)"; - private static final String DATA_FLOW_URN = "urn:li:dataFlow:(orchestrator,flowId,cluster)"; - private static final String ML_MODEL_URN = - "urn:li:mlModel:(urn:li:dataPlatform:sagemaker,trustmodel,PROD)"; - private static final String ML_MODEL_GROUP_URN = - "urn:li:mlModelGroup:(urn:li:dataPlatform:sagemaker,a-model-package-group,PROD)"; - private static final String ML_FEATURE_TABLE_URN = - "urn:li:mlFeatureTable:(urn:li:dataPlatform:feast,user_features)"; - private static final String ML_FEATURE_URN = "urn:li:mlFeature:(test,feature_1)"; - private static final List ENTITY_TYPES = - ImmutableList.of( - Constants.DATASET_ENTITY_NAME, - Constants.DASHBOARD_ENTITY_NAME, - Constants.CHART_ENTITY_NAME, - Constants.DATA_JOB_ENTITY_NAME, - Constants.DATA_FLOW_ENTITY_NAME, - Constants.ML_MODEL_ENTITY_NAME, - Constants.ML_MODEL_GROUP_ENTITY_NAME, - Constants.ML_FEATURE_TABLE_ENTITY_NAME, - Constants.ML_FEATURE_ENTITY_NAME); - private static final List ENTITY_URNS = - ImmutableList.of( - UrnUtils.getUrn(DATASET_URN), - UrnUtils.getUrn(DASHBOARD_URN), - UrnUtils.getUrn(CHART_URN), - UrnUtils.getUrn(DATA_JOB_URN), - UrnUtils.getUrn(DATA_FLOW_URN), - UrnUtils.getUrn(ML_MODEL_URN), - UrnUtils.getUrn(ML_MODEL_GROUP_URN), - UrnUtils.getUrn(ML_FEATURE_TABLE_URN), - UrnUtils.getUrn(ML_FEATURE_URN)); - - @Test - public void testExecuteNoExistingBrowsePaths() throws Exception { - final EntityService mockService = initMockService(); - final SearchService mockSearchService = initMockSearchService(); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(null); - - BackfillBrowsePathsV2Step backfillBrowsePathsV2Step = - new BackfillBrowsePathsV2Step(mockService, mockSearchService); - backfillBrowsePathsV2Step.execute(TestOperationContexts.systemContextNoSearchAuthorization()); - - Mockito.verify(mockSearchService, Mockito.times(9)) - .scrollAcrossEntities( - any(OperationContext.class), - any(), - Mockito.eq("*"), - any(Filter.class), - Mockito.eq(null), - Mockito.eq(null), - Mockito.eq("5m"), - Mockito.eq(5000)); - // Verify that 11 aspects are ingested, 2 for the upgrade request / result, 9 for ingesting 1 of - // each entity type - Mockito.verify(mockService, Mockito.times(11)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(), - Mockito.eq(false)); - } - - @Test - public void testDoesNotRunWhenAlreadyExecuted() throws Exception { - final EntityService mockService = mock(EntityService.class); - final SearchService mockSearchService = initMockSearchService(); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - com.linkedin.upgrade.DataHubUpgradeRequest upgradeRequest = - new com.linkedin.upgrade.DataHubUpgradeRequest().setVersion(VERSION); - Map upgradeRequestAspects = new HashMap<>(); - upgradeRequestAspects.put( - Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME, - new EnvelopedAspect().setValue(new Aspect(upgradeRequest.data()))); - EntityResponse response = - new EntityResponse().setAspects(new EnvelopedAspectMap(upgradeRequestAspects)); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(response); - - BackfillBrowsePathsV2Step backfillBrowsePathsV2Step = - new BackfillBrowsePathsV2Step(mockService, mockSearchService); - backfillBrowsePathsV2Step.execute(mock(OperationContext.class)); - - Mockito.verify(mockService, Mockito.times(0)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(AuditStamp.class), - Mockito.anyBoolean()); - } - - private EntityService initMockService() throws URISyntaxException { - final EntityService mockService = mock(EntityService.class); - - for (int i = 0; i < ENTITY_TYPES.size(); i++) { - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - any(), - Mockito.eq(ENTITY_URNS.get(i)), - Mockito.eq(Collections.singleton(CONTAINER_ASPECT_NAME)))) - .thenReturn(null); - } - - return mockService; - } - - private SearchService initMockSearchService() { - final SearchService mockSearchService = mock(SearchService.class); - - for (int i = 0; i < ENTITY_TYPES.size(); i++) { - Mockito.when( - mockSearchService.scrollAcrossEntities( - Mockito.any(OperationContext.class), - Mockito.eq(ImmutableList.of(ENTITY_TYPES.get(i))), - Mockito.eq("*"), - any(Filter.class), - Mockito.eq(null), - Mockito.eq(null), - Mockito.eq("5m"), - Mockito.eq(5000))) - .thenReturn( - new ScrollResult() - .setNumEntities(1) - .setEntities( - new SearchEntityArray(new SearchEntity().setEntity(ENTITY_URNS.get(i))))); - } - - return mockSearchService; - } -} diff --git a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/IngestEntityTypesStepTest.java b/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/IngestEntityTypesStepTest.java index a93bd8073a553..ab76f171aa7b1 100644 --- a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/IngestEntityTypesStepTest.java +++ b/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/IngestEntityTypesStepTest.java @@ -17,10 +17,18 @@ import io.datahubproject.test.metadata.context.TestOperationContexts; import org.jetbrains.annotations.NotNull; import org.mockito.Mockito; +import org.testng.annotations.BeforeTest; import org.testng.annotations.Test; public class IngestEntityTypesStepTest { + @BeforeTest + public static void beforeTest() { + PathSpecBasedSchemaAnnotationVisitor.class + .getClassLoader() + .setClassAssertionStatus(PathSpecBasedSchemaAnnotationVisitor.class.getName(), false); + } + @Test public void testExecuteTestEntityRegistry() throws Exception { EntityRegistry testEntityRegistry = getTestEntityRegistry(); diff --git a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStepTest.java b/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStepTest.java deleted file mode 100644 index 08f3dba12bd2a..0000000000000 --- a/metadata-service/factories/src/test/java/com/linkedin/metadata/boot/steps/UpgradeDefaultBrowsePathsStepTest.java +++ /dev/null @@ -1,406 +0,0 @@ -package com.linkedin.metadata.boot.steps; - -import static org.mockito.ArgumentMatchers.any; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; - -import com.google.common.collect.ImmutableList; -import com.linkedin.common.AuditStamp; -import com.linkedin.common.BrowsePaths; -import com.linkedin.common.urn.Urn; -import com.linkedin.common.urn.UrnUtils; -import com.linkedin.data.template.RecordTemplate; -import com.linkedin.data.template.StringArray; -import com.linkedin.entity.Aspect; -import com.linkedin.entity.EntityResponse; -import com.linkedin.entity.EnvelopedAspect; -import com.linkedin.entity.EnvelopedAspectMap; -import com.linkedin.metadata.Constants; -import com.linkedin.metadata.aspect.patch.template.AspectTemplateEngine; -import com.linkedin.metadata.entity.EntityService; -import com.linkedin.metadata.entity.ListResult; -import com.linkedin.metadata.models.AspectSpec; -import com.linkedin.metadata.models.EntitySpec; -import com.linkedin.metadata.models.EntitySpecBuilder; -import com.linkedin.metadata.models.EventSpec; -import com.linkedin.metadata.models.registry.EntityRegistry; -import com.linkedin.metadata.query.ExtraInfo; -import com.linkedin.metadata.query.ExtraInfoArray; -import com.linkedin.metadata.query.ListResultMetadata; -import com.linkedin.metadata.search.utils.BrowsePathUtils; -import com.linkedin.metadata.snapshot.Snapshot; -import com.linkedin.mxe.MetadataChangeProposal; -import io.datahubproject.metadata.context.OperationContext; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.stream.Collectors; -import javax.annotation.Nonnull; -import javax.annotation.Nullable; -import org.jetbrains.annotations.NotNull; -import org.mockito.Mockito; -import org.testng.annotations.Test; - -public class UpgradeDefaultBrowsePathsStepTest { - - private static final String VERSION_1 = "1"; - private static final String UPGRADE_URN = - String.format( - "urn:li:%s:%s", - Constants.DATA_HUB_UPGRADE_ENTITY_NAME, "upgrade-default-browse-paths-step"); - - @Test - public void testExecuteNoExistingBrowsePaths() throws Exception { - - final EntityService mockService = Mockito.mock(EntityService.class); - final EntityRegistry registry = new TestEntityRegistry(); - final OperationContext mockContext = mock(OperationContext.class); - when(mockContext.getEntityRegistry()).thenReturn(registry); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(null); - - final List browsePaths1 = Collections.emptyList(); - - Mockito.when( - mockService.listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000))) - .thenReturn( - new ListResult<>( - browsePaths1, - new ListResultMetadata().setExtraInfos(new ExtraInfoArray(Collections.emptyList())), - 0, - false, - 0, - 0, - 2)); - initMockServiceOtherEntities(mockService); - - UpgradeDefaultBrowsePathsStep upgradeDefaultBrowsePathsStep = - new UpgradeDefaultBrowsePathsStep(mockService); - upgradeDefaultBrowsePathsStep.execute(mockContext); - - Mockito.verify(mockService, Mockito.times(1)) - .listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000)); - // Verify that 4 aspects are ingested, 2 for the upgrade request / result, but none for - // ingesting - Mockito.verify(mockService, Mockito.times(2)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(), - Mockito.eq(false)); - } - - @Test - public void testExecuteFirstTime() throws Exception { - - Urn testUrn1 = - UrnUtils.getUrn("urn:li:dataset:(urn:li:dataPlatform:kafka,SampleKafkaDataset1,PROD)"); - Urn testUrn2 = - UrnUtils.getUrn("urn:li:dataset:(urn:li:dataPlatform:kafka,SampleKafkaDataset2,PROD)"); - - final EntityService mockService = Mockito.mock(EntityService.class); - final EntityRegistry registry = new TestEntityRegistry(); - final OperationContext mockContext = mock(OperationContext.class); - when(mockContext.getEntityRegistry()).thenReturn(registry); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(null); - final List browsePaths1 = - ImmutableList.of( - new BrowsePaths() - .setPaths( - new StringArray( - ImmutableList.of( - BrowsePathUtils.getLegacyDefaultBrowsePath(testUrn1, registry)))), - new BrowsePaths() - .setPaths( - new StringArray( - ImmutableList.of( - BrowsePathUtils.getLegacyDefaultBrowsePath(testUrn2, registry))))); - - final List extraInfos1 = - ImmutableList.of( - new ExtraInfo() - .setUrn(testUrn1) - .setVersion(0L) - .setAudit( - new AuditStamp().setActor(UrnUtils.getUrn("urn:li:corpuser:test")).setTime(0L)), - new ExtraInfo() - .setUrn(testUrn2) - .setVersion(0L) - .setAudit( - new AuditStamp() - .setActor(UrnUtils.getUrn("urn:li:corpuser:test")) - .setTime(0L))); - - Mockito.when( - mockService.listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000))) - .thenReturn( - new ListResult<>( - browsePaths1, - new ListResultMetadata().setExtraInfos(new ExtraInfoArray(extraInfos1)), - 2, - false, - 2, - 2, - 2)); - initMockServiceOtherEntities(mockService); - - UpgradeDefaultBrowsePathsStep upgradeDefaultBrowsePathsStep = - new UpgradeDefaultBrowsePathsStep(mockService); - upgradeDefaultBrowsePathsStep.execute(mockContext); - - Mockito.verify(mockService, Mockito.times(1)) - .listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000)); - // Verify that 4 aspects are ingested, 2 for the upgrade request / result and 2 for the browse - // pahts - Mockito.verify(mockService, Mockito.times(4)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(), - Mockito.eq(false)); - } - - @Test - public void testDoesNotRunWhenBrowsePathIsNotQualified() throws Exception { - // Test for browse paths that are not ingested - Urn testUrn3 = - UrnUtils.getUrn( - "urn:li:dataset:(urn:li:dataPlatform:kafka,SampleKafkaDataset3,PROD)"); // Do not - // migrate - Urn testUrn4 = - UrnUtils.getUrn( - "urn:li:dataset:(urn:li:dataPlatform:kafka,SampleKafkaDataset4,PROD)"); // Do not - // migrate - - final EntityService mockService = Mockito.mock(EntityService.class); - final EntityRegistry registry = new TestEntityRegistry(); - final OperationContext mockContext = mock(OperationContext.class); - when(mockContext.getEntityRegistry()).thenReturn(registry); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(null); - - final List browsePaths2 = - ImmutableList.of( - new BrowsePaths() - .setPaths( - new StringArray( - ImmutableList.of( - BrowsePathUtils.getDefaultBrowsePath(testUrn3, registry, '.')))), - new BrowsePaths() - .setPaths( - new StringArray( - ImmutableList.of( - BrowsePathUtils.getLegacyDefaultBrowsePath(testUrn4, registry), - BrowsePathUtils.getDefaultBrowsePath(testUrn4, registry, '.'))))); - - final List extraInfos2 = - ImmutableList.of( - new ExtraInfo() - .setUrn(testUrn3) - .setVersion(0L) - .setAudit( - new AuditStamp().setActor(UrnUtils.getUrn("urn:li:corpuser:test")).setTime(0L)), - new ExtraInfo() - .setUrn(testUrn4) - .setVersion(0L) - .setAudit( - new AuditStamp() - .setActor(UrnUtils.getUrn("urn:li:corpuser:test")) - .setTime(0L))); - - Mockito.when( - mockService.listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000))) - .thenReturn( - new ListResult<>( - browsePaths2, - new ListResultMetadata().setExtraInfos(new ExtraInfoArray(extraInfos2)), - 2, - false, - 2, - 2, - 2)); - initMockServiceOtherEntities(mockService); - - UpgradeDefaultBrowsePathsStep upgradeDefaultBrowsePathsStep = - new UpgradeDefaultBrowsePathsStep(mockService); - upgradeDefaultBrowsePathsStep.execute(mockContext); - - Mockito.verify(mockService, Mockito.times(1)) - .listLatestAspects( - any(OperationContext.class), - Mockito.eq(Constants.DATASET_ENTITY_NAME), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000)); - // Verify that 2 aspects are ingested, only those for the upgrade step - Mockito.verify(mockService, Mockito.times(2)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(), - Mockito.eq(false)); - } - - @Test - public void testDoesNotRunWhenAlreadyExecuted() throws Exception { - final EntityService mockService = Mockito.mock(EntityService.class); - final OperationContext mockContext = mock(OperationContext.class); - when(mockContext.getEntityRegistry()).thenReturn(mock(EntityRegistry.class)); - - final Urn upgradeEntityUrn = Urn.createFromString(UPGRADE_URN); - com.linkedin.upgrade.DataHubUpgradeRequest upgradeRequest = - new com.linkedin.upgrade.DataHubUpgradeRequest().setVersion(VERSION_1); - Map upgradeRequestAspects = new HashMap<>(); - upgradeRequestAspects.put( - Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME, - new EnvelopedAspect().setValue(new Aspect(upgradeRequest.data()))); - EntityResponse response = - new EntityResponse().setAspects(new EnvelopedAspectMap(upgradeRequestAspects)); - Mockito.when( - mockService.getEntityV2( - any(OperationContext.class), - Mockito.eq(Constants.DATA_HUB_UPGRADE_ENTITY_NAME), - Mockito.eq(upgradeEntityUrn), - Mockito.eq(Collections.singleton(Constants.DATA_HUB_UPGRADE_REQUEST_ASPECT_NAME)))) - .thenReturn(response); - - UpgradeDefaultBrowsePathsStep step = new UpgradeDefaultBrowsePathsStep(mockService); - step.execute(mockContext); - - Mockito.verify(mockService, Mockito.times(0)) - .ingestProposal( - any(OperationContext.class), - any(MetadataChangeProposal.class), - any(AuditStamp.class), - Mockito.anyBoolean()); - } - - private void initMockServiceOtherEntities(EntityService mockService) { - List skippedEntityTypes = - ImmutableList.of( - Constants.DASHBOARD_ENTITY_NAME, - Constants.CHART_ENTITY_NAME, - Constants.DATA_FLOW_ENTITY_NAME, - Constants.DATA_JOB_ENTITY_NAME); - for (String entityType : skippedEntityTypes) { - Mockito.when( - mockService.listLatestAspects( - any(OperationContext.class), - Mockito.eq(entityType), - Mockito.eq(Constants.BROWSE_PATHS_ASPECT_NAME), - Mockito.eq(0), - Mockito.eq(5000))) - .thenReturn( - new ListResult<>( - Collections.emptyList(), - new ListResultMetadata() - .setExtraInfos(new ExtraInfoArray(Collections.emptyList())), - 0, - false, - 0, - 0, - 0)); - } - } - - public static class TestEntityRegistry implements EntityRegistry { - - private final Map entityNameToSpec; - - public TestEntityRegistry() { - entityNameToSpec = - new EntitySpecBuilder(EntitySpecBuilder.AnnotationExtractionMode.IGNORE_ASPECT_FIELDS) - .buildEntitySpecs(new Snapshot().schema()).stream() - .collect(Collectors.toMap(spec -> spec.getName().toLowerCase(), spec -> spec)); - } - - @Nonnull - @Override - public EntitySpec getEntitySpec(@Nonnull final String entityName) { - String lowercaseEntityName = entityName.toLowerCase(); - if (!entityNameToSpec.containsKey(lowercaseEntityName)) { - throw new IllegalArgumentException( - String.format("Failed to find entity with name %s in EntityRegistry", entityName)); - } - return entityNameToSpec.get(lowercaseEntityName); - } - - @Nullable - @Override - public EventSpec getEventSpec(@Nonnull String eventName) { - return null; - } - - @Nonnull - @Override - public Map getEntitySpecs() { - return entityNameToSpec; - } - - @NotNull - @Override - public Map getAspectSpecs() { - return new HashMap<>(); - } - - @Nonnull - @Override - public Map getEventSpecs() { - return Collections.emptyMap(); - } - - @NotNull - @Override - public AspectTemplateEngine getAspectTemplateEngine() { - return new AspectTemplateEngine(); - } - } -} From 7340ec79f38b868bcd703cf0c9e06f201f775a59 Mon Sep 17 00:00:00 2001 From: david-leifker <114954101+david-leifker@users.noreply.github.com> Date: Tue, 1 Oct 2024 07:30:36 -0500 Subject: [PATCH 2/9] docs(search): example of entity weighting (#11511) --- docs/how/search.md | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/docs/how/search.md b/docs/how/search.md index 5c1ba266ee2ae..4df5e7c1984d5 100644 --- a/docs/how/search.md +++ b/docs/how/search.md @@ -398,6 +398,32 @@ queryConfigurations: boost_mode: multiply ``` +##### Example 4: Entity Ranking + +Alter the ranking of entities. For example, chart vs dashboard, you may want the dashboard +to appear above charts. This can be done using the following function score and leverages a prefix match on the entity type +of the URN. Depending on the entity the weight may have to be adjusted based on your data and the entities +involved since often multiple field matches may shift weight towards one entity vs another. + +```yaml +queryConfigurations: + - queryRegex: .* + + simpleQuery: true + prefixMatchQuery: true + exactMatchQuery: true + + functionScore: + functions: + - filter: + prefix: + urn: + value: 'urn:li:dashboard:' + weight: 1.5 + score_mode: multiply + boost_mode: multiply +``` + ### Search Autocomplete Configuration Similar to the options provided in the previous section for search configuration, there are autocomplete specific options From a0787684de29d67dc7d6049ec7025d861e26522e Mon Sep 17 00:00:00 2001 From: deepgarg-visa <149145061+deepgarg-visa@users.noreply.github.com> Date: Tue, 1 Oct 2024 19:27:08 +0530 Subject: [PATCH 3/9] =?UTF-8?q?feat(businessattribute):=20filter=20schema?= =?UTF-8?q?=20rows=20on=20business-attribute=20pro=E2=80=A6=20(#11502)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Schema/__tests__/filterSchemaRows.test.ts | 96 +++++++++++++++++++ .../Dataset/Schema/utils/filterSchemaRows.ts | 25 ++++- 2 files changed, 119 insertions(+), 2 deletions(-) diff --git a/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/__tests__/filterSchemaRows.test.ts b/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/__tests__/filterSchemaRows.test.ts index 27c0af87fc833..87fca3b898c83 100644 --- a/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/__tests__/filterSchemaRows.test.ts +++ b/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/__tests__/filterSchemaRows.test.ts @@ -235,4 +235,100 @@ describe('filterSchemaRows', () => { expect(filteredRows).toMatchObject([{ fieldPath: 'shipment' }]); expect(expandedRowsFromFilter).toMatchObject(new Set()); }); + + it('should properly filter schema rows based on business attribute properties description', () => { + const rowsWithSchemaFieldEntity = [ + { + fieldPath: 'customer', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { properties: { description: 'customer description' } }, + }, + }, + }, + }, + { + fieldPath: 'testing', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { properties: { description: 'testing description' } }, + }, + }, + }, + }, + { + fieldPath: 'shipment', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { properties: { description: 'shipment description' } }, + }, + }, + }, + }, + ] as SchemaField[]; + const filterText = 'testing description'; + const editableSchemaMetadata = { editableSchemaFieldInfo: [] }; + const { filteredRows, expandedRowsFromFilter } = filterSchemaRows( + rowsWithSchemaFieldEntity, + editableSchemaMetadata, + filterText, + testEntityRegistry, + ); + + expect(filteredRows).toMatchObject([{ fieldPath: 'testing' }]); + expect(expandedRowsFromFilter).toMatchObject(new Set()); + }); + + it('should properly filter schema rows based on business attribute properties tags', () => { + const rowsWithSchemaFieldEntity = [ + { + fieldPath: 'customer', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { properties: { tags: { tags: [{ tag: sampleTag }] } } }, + }, + }, + }, + }, + { + fieldPath: 'testing', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { + properties: { tags: { tags: [{ tag: { properties: { name: 'otherTag' } } }] } }, + }, + }, + }, + }, + }, + { + fieldPath: 'shipment', + schemaFieldEntity: { + businessAttributes: { + businessAttribute: { + businessAttribute: { + properties: { tags: { tags: [{ tag: { properties: { name: 'anotherTag' } } }] } }, + }, + }, + }, + }, + }, + ] as SchemaField[]; + const filterText = sampleTag.properties.name; + const editableSchemaMetadata = { editableSchemaFieldInfo: [] }; + const { filteredRows, expandedRowsFromFilter } = filterSchemaRows( + rowsWithSchemaFieldEntity, + editableSchemaMetadata, + filterText, + testEntityRegistry, + ); + + expect(filteredRows).toMatchObject([{ fieldPath: 'customer' }]); + expect(expandedRowsFromFilter).toMatchObject(new Set()); + }); }); diff --git a/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/utils/filterSchemaRows.ts b/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/utils/filterSchemaRows.ts index 96505e1bee785..53b76d53f886a 100644 --- a/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/utils/filterSchemaRows.ts +++ b/datahub-web-react/src/app/entity/shared/tabs/Dataset/Schema/utils/filterSchemaRows.ts @@ -16,6 +16,25 @@ function matchesTagsOrTermsOrDescription(field: SchemaField, filterText: string, ); } +function matchesBusinessAttributesProperties(field: SchemaField, filterText: string, entityRegistry: EntityRegistry) { + if (!field.schemaFieldEntity?.businessAttributes) return false; + const businessAttributeProperties = + field.schemaFieldEntity?.businessAttributes?.businessAttribute?.businessAttribute?.properties; + return ( + businessAttributeProperties?.description?.toLocaleLowerCase().includes(filterText) || + businessAttributeProperties?.name?.toLocaleLowerCase().includes(filterText) || + businessAttributeProperties?.glossaryTerms?.terms?.find((termAssociation) => + entityRegistry + .getDisplayName(EntityType.GlossaryTerm, termAssociation.term) + .toLocaleLowerCase() + .includes(filterText), + ) || + businessAttributeProperties?.tags?.tags?.find((tagAssociation) => + entityRegistry.getDisplayName(EntityType.Tag, tagAssociation.tag).toLocaleLowerCase().includes(filterText), + ) + ); +} + // returns list of fieldPaths for fields that have Terms or Tags or Descriptions matching the filterText function getFilteredFieldPathsByMetadata(editableSchemaMetadata: any, entityRegistry, filterText) { return ( @@ -56,7 +75,8 @@ export function filterSchemaRows( if ( matchesFieldName(row.fieldPath, formattedFilterText) || matchesEditableTagsOrTermsOrDescription(row, filteredFieldPathsByEditableMetadata) || - matchesTagsOrTermsOrDescription(row, formattedFilterText, entityRegistry) // non-editable tags, terms and description + matchesTagsOrTermsOrDescription(row, formattedFilterText, entityRegistry) || // non-editable tags, terms and description + matchesBusinessAttributesProperties(row, formattedFilterText, entityRegistry) ) { finalFieldPaths.add(row.fieldPath); } @@ -65,7 +85,8 @@ export function filterSchemaRows( if ( matchesFieldName(fieldName, formattedFilterText) || matchesEditableTagsOrTermsOrDescription(row, filteredFieldPathsByEditableMetadata) || - matchesTagsOrTermsOrDescription(row, formattedFilterText, entityRegistry) // non-editable tags, terms and description + matchesTagsOrTermsOrDescription(row, formattedFilterText, entityRegistry) || // non-editable tags, terms and description + matchesBusinessAttributesProperties(row, formattedFilterText, entityRegistry) ) { // if we match specifically on this field (not just its parent), add and expand all parents splitFieldPath.reduce((previous, current) => { From 67d711605505adc5fe18bad71cd6baabeea48323 Mon Sep 17 00:00:00 2001 From: sid-acryl <155424659+sid-acryl@users.noreply.github.com> Date: Tue, 1 Oct 2024 23:56:00 +0530 Subject: [PATCH 4/9] fix(ingest/lookml): missing lineage for looker template -- if prod (#11426) --- .../source/looker/looker_dataclasses.py | 21 ++++++++++++++++--- .../source/looker/looker_file_loader.py | 10 ++++----- .../source/looker/looker_template_language.py | 20 +++++++++++++++++- .../source/looker/lookml_concept_context.py | 5 +++-- .../ingestion/source/looker/lookml_source.py | 17 ++++++++++++--- .../ingestion/source/looker/view_upstream.py | 10 +++++++-- .../tests/integration/lookml/test_lookml.py | 9 ++++++-- 7 files changed, 73 insertions(+), 19 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_dataclasses.py b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_dataclasses.py index adaa3c4875450..7e23079156b62 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_dataclasses.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_dataclasses.py @@ -4,11 +4,14 @@ from dataclasses import dataclass from typing import Dict, List, Optional, Set -from datahub.ingestion.source.looker.lkml_patched import load_lkml from datahub.ingestion.source.looker.looker_connection import LookerConnectionDefinition +from datahub.ingestion.source.looker.looker_template_language import ( + load_and_preprocess_file, +) from datahub.ingestion.source.looker.lookml_config import ( _BASE_PROJECT_NAME, _EXPLORE_FILE_EXTENSION, + LookMLSourceConfig, LookMLSourceReport, ) @@ -43,6 +46,7 @@ def from_looker_dict( root_project_name: Optional[str], base_projects_folders: Dict[str, pathlib.Path], path: str, + source_config: LookMLSourceConfig, reporter: LookMLSourceReport, ) -> "LookerModel": logger.debug(f"Loading model from {path}") @@ -54,6 +58,7 @@ def from_looker_dict( root_project_name, base_projects_folders, path, + source_config, reporter, seen_so_far=set(), traversal_path=pathlib.Path(path).stem, @@ -68,7 +73,10 @@ def from_looker_dict( ] for included_file in explore_files: try: - parsed = load_lkml(included_file) + parsed = load_and_preprocess_file( + path=included_file, + source_config=source_config, + ) included_explores = parsed.get("explores", []) explores.extend(included_explores) except Exception as e: @@ -94,6 +102,7 @@ def resolve_includes( root_project_name: Optional[str], base_projects_folder: Dict[str, pathlib.Path], path: str, + source_config: LookMLSourceConfig, reporter: LookMLSourceReport, seen_so_far: Set[str], traversal_path: str = "", # a cosmetic parameter to aid debugging @@ -206,7 +215,10 @@ def resolve_includes( f"Will be loading {included_file}, traversed here via {traversal_path}" ) try: - parsed = load_lkml(included_file) + parsed = load_and_preprocess_file( + path=included_file, + source_config=source_config, + ) seen_so_far.add(included_file) if "includes" in parsed: # we have more includes to resolve! resolved.extend( @@ -216,6 +228,7 @@ def resolve_includes( root_project_name, base_projects_folder, included_file, + source_config, reporter, seen_so_far, traversal_path=traversal_path @@ -259,6 +272,7 @@ def from_looker_dict( root_project_name: Optional[str], base_projects_folder: Dict[str, pathlib.Path], raw_file_content: str, + source_config: LookMLSourceConfig, reporter: LookMLSourceReport, ) -> "LookerViewFile": logger.debug(f"Loading view file at {absolute_file_path}") @@ -272,6 +286,7 @@ def from_looker_dict( root_project_name, base_projects_folder, absolute_file_path, + source_config, reporter, seen_so_far=seen_so_far, ) diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_file_loader.py b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_file_loader.py index 52ebcdde06a27..f894c96debc54 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_file_loader.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_file_loader.py @@ -3,11 +3,10 @@ from dataclasses import replace from typing import Dict, Optional -from datahub.ingestion.source.looker.lkml_patched import load_lkml from datahub.ingestion.source.looker.looker_config import LookerConnectionDefinition from datahub.ingestion.source.looker.looker_dataclasses import LookerViewFile from datahub.ingestion.source.looker.looker_template_language import ( - process_lookml_template_language, + load_and_preprocess_file, ) from datahub.ingestion.source.looker.lookml_config import ( _EXPLORE_FILE_EXTENSION, @@ -72,10 +71,8 @@ def _load_viewfile( try: logger.debug(f"Loading viewfile {path}") - parsed = load_lkml(path) - - process_lookml_template_language( - view_lkml_file_dict=parsed, + parsed = load_and_preprocess_file( + path=path, source_config=self.source_config, ) @@ -86,6 +83,7 @@ def _load_viewfile( root_project_name=self._root_project_name, base_projects_folder=self._base_projects_folder, raw_file_content=raw_file_content, + source_config=self.source_config, reporter=reporter, ) logger.debug(f"adding viewfile for path {path} to the cache") diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_template_language.py b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_template_language.py index 04f9ec081ee68..1e60c08fe00c2 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_template_language.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_template_language.py @@ -1,12 +1,14 @@ import logging +import pathlib import re from abc import ABC, abstractmethod -from typing import Any, ClassVar, Dict, List, Optional, Set +from typing import Any, ClassVar, Dict, List, Optional, Set, Union from deepmerge import always_merger from liquid import Undefined from liquid.exceptions import LiquidSyntaxError +from datahub.ingestion.source.looker.lkml_patched import load_lkml from datahub.ingestion.source.looker.looker_constant import ( DATAHUB_TRANSFORMED_SQL, DATAHUB_TRANSFORMED_SQL_TABLE_NAME, @@ -390,6 +392,7 @@ def process_lookml_template_language( source_config: LookMLSourceConfig, view_lkml_file_dict: dict, ) -> None: + if "views" not in view_lkml_file_dict: return @@ -416,3 +419,18 @@ def process_lookml_template_language( ) view_lkml_file_dict["views"] = transformed_views + + +def load_and_preprocess_file( + path: Union[str, pathlib.Path], + source_config: LookMLSourceConfig, +) -> dict: + + parsed = load_lkml(path) + + process_lookml_template_language( + view_lkml_file_dict=parsed, + source_config=source_config, + ) + + return parsed diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_concept_context.py b/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_concept_context.py index bf24f4b84679b..ce4a242027e11 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_concept_context.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_concept_context.py @@ -365,8 +365,9 @@ def sql_table_name(self) -> str: return sql_table_name.lower() def datahub_transformed_sql_table_name(self) -> str: - table_name: Optional[str] = self.raw_view.get( - "datahub_transformed_sql_table_name" + # This field might be present in parent view of current view + table_name: Optional[str] = self.get_including_extends( + field="datahub_transformed_sql_table_name" ) if not table_name: diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_source.py b/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_source.py index b00291caabbf6..e4d8dd19fb791 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_source.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/lookml_source.py @@ -29,7 +29,6 @@ DatasetSubTypes, ) from datahub.ingestion.source.git.git_import import GitClone -from datahub.ingestion.source.looker.lkml_patched import load_lkml from datahub.ingestion.source.looker.looker_common import ( CORPUSER_DATAHUB, LookerExplore, @@ -45,6 +44,9 @@ get_connection_def_based_on_connection_string, ) from datahub.ingestion.source.looker.looker_lib_wrapper import LookerAPI +from datahub.ingestion.source.looker.looker_template_language import ( + load_and_preprocess_file, +) from datahub.ingestion.source.looker.looker_view_id_cache import ( LookerModel, LookerViewFileLoader, @@ -311,13 +313,19 @@ def __init__(self, config: LookMLSourceConfig, ctx: PipelineContext): def _load_model(self, path: str) -> LookerModel: logger.debug(f"Loading model from file {path}") - parsed = load_lkml(path) + + parsed = load_and_preprocess_file( + path=path, + source_config=self.source_config, + ) + looker_model = LookerModel.from_looker_dict( parsed, _BASE_PROJECT_NAME, self.source_config.project_name, self.base_projects_folder, path, + self.source_config, self.reporter, ) return looker_model @@ -495,7 +503,10 @@ def get_project_name(self, model_name: str) -> str: def get_manifest_if_present(self, folder: pathlib.Path) -> Optional[LookerManifest]: manifest_file = folder / "manifest.lkml" if manifest_file.exists(): - manifest_dict = load_lkml(manifest_file) + + manifest_dict = load_and_preprocess_file( + path=manifest_file, source_config=self.source_config + ) manifest = LookerManifest( project_name=manifest_dict.get("project_name"), diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/view_upstream.py b/metadata-ingestion/src/datahub/ingestion/source/looker/view_upstream.py index de1022b5482ce..057dbca428184 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/view_upstream.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/view_upstream.py @@ -154,6 +154,7 @@ def _generate_fully_qualified_name( sql_table_name: str, connection_def: LookerConnectionDefinition, reporter: LookMLSourceReport, + view_name: str, ) -> str: """Returns a fully qualified dataset name, resolved through a connection definition. Input sql_table_name can be in three forms: table, db.table, db.schema.table""" @@ -192,7 +193,7 @@ def _generate_fully_qualified_name( reporter.report_warning( title="Malformed Table Name", message="Table name has more than 3 parts.", - context=f"Table Name: {sql_table_name}", + context=f"view-name: {view_name}, table-name: {sql_table_name}", ) return sql_table_name.lower() @@ -280,10 +281,13 @@ def __get_upstream_dataset_urn(self) -> List[Urn]: return [] if sql_parsing_result.debug_info.table_error is not None: + logger.debug( + f"view-name={self.view_context.name()}, sql_query={self.get_sql_query()}" + ) self.reporter.report_warning( title="Table Level Lineage Missing", message="Error in parsing derived sql", - context=f"View-name: {self.view_context.name()}", + context=f"view-name: {self.view_context.name()}, platform: {self.view_context.view_connection.platform}", exc=sql_parsing_result.debug_info.table_error, ) return [] @@ -530,6 +534,7 @@ def __get_upstream_dataset_urn(self) -> Urn: sql_table_name=self.view_context.datahub_transformed_sql_table_name(), connection_def=self.view_context.view_connection, reporter=self.view_context.reporter, + view_name=self.view_context.name(), ) self.upstream_dataset_urn = make_dataset_urn_with_platform_instance( @@ -586,6 +591,7 @@ def __get_upstream_dataset_urn(self) -> List[Urn]: self.view_context.datahub_transformed_sql_table_name(), self.view_context.view_connection, self.view_context.reporter, + self.view_context.name(), ), base_folder_path=self.view_context.base_folder_path, looker_view_id_cache=self.looker_view_id_cache, diff --git a/metadata-ingestion/tests/integration/lookml/test_lookml.py b/metadata-ingestion/tests/integration/lookml/test_lookml.py index a5d838cb16d73..e4eb564e3e86b 100644 --- a/metadata-ingestion/tests/integration/lookml/test_lookml.py +++ b/metadata-ingestion/tests/integration/lookml/test_lookml.py @@ -2,6 +2,7 @@ import pathlib from typing import Any, List from unittest import mock +from unittest.mock import MagicMock import pydantic import pytest @@ -14,13 +15,13 @@ from datahub.ingestion.source.file import read_metadata_file from datahub.ingestion.source.looker.looker_template_language import ( SpecialVariable, + load_and_preprocess_file, resolve_liquid_variable, ) from datahub.ingestion.source.looker.lookml_source import ( LookerModel, LookerRefinementResolver, LookMLSourceConfig, - load_lkml, ) from datahub.metadata.schema_classes import ( DatasetSnapshotClass, @@ -870,7 +871,11 @@ def test_manifest_parser(pytestconfig: pytest.Config) -> None: test_resources_dir = pytestconfig.rootpath / "tests/integration/lookml" manifest_file = test_resources_dir / "lkml_manifest_samples/complex-manifest.lkml" - manifest = load_lkml(manifest_file) + manifest = load_and_preprocess_file( + path=manifest_file, + source_config=MagicMock(), + ) + assert manifest From 660fbf8e57cac5846095c1f318ac98fe7ad56b1c Mon Sep 17 00:00:00 2001 From: sagar-salvi-apptware <159135491+sagar-salvi-apptware@users.noreply.github.com> Date: Wed, 2 Oct 2024 00:09:07 +0530 Subject: [PATCH 5/9] fix(ingestion/transformer): Add container support for ownership and domains (#11375) --- .../docs/transformer/dataset_transformer.md | 62 ++ .../transformer/add_dataset_ownership.py | 59 +- .../ingestion/transformer/dataset_domain.py | 66 +- .../tests/unit/test_transform_dataset.py | 566 ++++++++++++++++++ 4 files changed, 750 insertions(+), 3 deletions(-) diff --git a/metadata-ingestion/docs/transformer/dataset_transformer.md b/metadata-ingestion/docs/transformer/dataset_transformer.md index 03a224bcf7da4..d48c6d2c1ab5b 100644 --- a/metadata-ingestion/docs/transformer/dataset_transformer.md +++ b/metadata-ingestion/docs/transformer/dataset_transformer.md @@ -197,9 +197,12 @@ transformers: | `ownership_type` | | string | "DATAOWNER" | ownership type of the owners (either as enum or ownership type urn) | | `replace_existing` | | boolean | `false` | Whether to remove owners from entity sent by ingestion source. | | `semantics` | | enum | `OVERWRITE` | Whether to OVERWRITE or PATCH the entity present on DataHub GMS. | +| `is_container` | | bool | `false` | Whether to also consider a container or not. If true, then ownership will be attached to both the dataset and its container. | let’s suppose we’d like to append a series of users who we know to own a different dataset from a data source but aren't detected during normal ingestion. To do so, we can use the `pattern_add_dataset_ownership` module that’s included in the ingestion framework. This will match the pattern to `urn` of the dataset and assign the respective owners. +If the is_container field is set to true, the module will not only attach the ownerships to the matching datasets but will also find and attach containers associated with those datasets. This means that both the datasets and their containers will be associated with the specified owners. + The config, which we’d append to our ingestion recipe YAML, would look like this: ```yaml @@ -251,6 +254,35 @@ The config, which we’d append to our ingestion recipe YAML, would look like th ".*example2.*": ["urn:li:corpuser:username2"] ownership_type: "PRODUCER" ``` +- Add owner to dataset and its containers + ```yaml + transformers: + - type: "pattern_add_dataset_ownership" + config: + is_container: true + replace_existing: true # false is default behaviour + semantics: PATCH / OVERWRITE # Based on user + owner_pattern: + rules: + ".*example1.*": ["urn:li:corpuser:username1"] + ".*example2.*": ["urn:li:corpuser:username2"] + ownership_type: "PRODUCER" + ``` +⚠️ Warning: +When working with two datasets in the same container but with different owners, all owners will be added for that dataset containers. + +For example: +```yaml +transformers: + - type: "pattern_add_dataset_ownership" + config: + is_container: true + owner_pattern: + rules: + ".*example1.*": ["urn:li:corpuser:username1"] + ".*example2.*": ["urn:li:corpuser:username2"] +``` +If example1 and example2 are in the same container, then both urns urn:li:corpuser:username1 and urn:li:corpuser:username2 will be added for respective dataset containers. ## Simple Remove Dataset ownership If we wanted to clear existing owners sent by ingestion source we can use the `simple_remove_dataset_ownership` transformer which removes all owners sent by the ingestion source. @@ -1074,10 +1106,13 @@ transformers: | `domain_pattern` | ✅ | map[regx, list[union[urn, str]] | | dataset urn with regular expression and list of simple domain name or domain urn need to be apply on matching dataset urn. | | `replace_existing` | | boolean | `false` | Whether to remove domains from entity sent by ingestion source. | | `semantics` | | enum | `OVERWRITE` | Whether to OVERWRITE or PATCH the entity present on DataHub GMS. | +| `is_container` | | bool | `false` | Whether to also consider a container or not. If true, then domains will be attached to both the dataset and its container. | Let’s suppose we’d like to append a series of domain to specific datasets. To do so, we can use the pattern_add_dataset_domain transformer that’s included in the ingestion framework. This will match the regex pattern to urn of the dataset and assign the respective domain urns given in the array. +If the is_container field is set to true, the module will not only attach the domains to the matching datasets but will also find and attach containers associated with those datasets. This means that both the datasets and their containers will be associated with the specified owners. + The config, which we’d append to our ingestion recipe YAML, would look like this: Here we can set domain list to either urn (i.e. urn:li:domain:hr) or simple domain name (i.e. hr) in both of the cases domain should be provisioned on DataHub GMS @@ -1129,6 +1164,33 @@ in both of the cases domain should be provisioned on DataHub GMS 'urn:li:dataset:\(urn:li:dataPlatform:postgres,postgres\.public\.t.*': ["urn:li:domain:finance"] ``` +- Add domains to dataset and its containers + ```yaml + transformers: + - type: "pattern_add_dataset_domain" + config: + is_container: true + semantics: PATCH / OVERWRITE # Based on user + domain_pattern: + rules: + 'urn:li:dataset:\(urn:li:dataPlatform:postgres,postgres\.public\.n.*': ["hr"] + 'urn:li:dataset:\(urn:li:dataPlatform:postgres,postgres\.public\.t.*': ["urn:li:domain:finance"] + ``` +⚠️ Warning: +When working with two datasets in the same container but with different domains, all domains will be added for that dataset containers. + +For example: +```yaml +transformers: + - type: "pattern_add_dataset_domain" + config: + is_container: true + domain_pattern: + rules: + ".*example1.*": ["hr"] + ".*example2.*": ["urn:li:domain:finance"] +``` +If example1 and example2 are in the same container, then both domains hr and finance will be added for respective dataset containers. ## Domain Mapping Based on Tags diff --git a/metadata-ingestion/src/datahub/ingestion/transformer/add_dataset_ownership.py b/metadata-ingestion/src/datahub/ingestion/transformer/add_dataset_ownership.py index 5112a443768db..54be2e5fac1e3 100644 --- a/metadata-ingestion/src/datahub/ingestion/transformer/add_dataset_ownership.py +++ b/metadata-ingestion/src/datahub/ingestion/transformer/add_dataset_ownership.py @@ -1,4 +1,5 @@ -from typing import Callable, List, Optional, cast +import logging +from typing import Callable, Dict, List, Optional, Union, cast import datahub.emitter.mce_builder as builder from datahub.configuration.common import ( @@ -9,16 +10,22 @@ ) from datahub.configuration.import_resolver import pydantic_resolve_key from datahub.emitter.mce_builder import Aspect +from datahub.emitter.mcp import MetadataChangeProposalWrapper from datahub.ingestion.api.common import PipelineContext from datahub.ingestion.graph.client import DataHubGraph from datahub.ingestion.transformer.dataset_transformer import ( DatasetOwnershipTransformer, ) from datahub.metadata.schema_classes import ( + BrowsePathsV2Class, + MetadataChangeProposalClass, OwnerClass, OwnershipClass, OwnershipTypeClass, ) +from datahub.specific.dashboard import DashboardPatchBuilder + +logger = logging.getLogger(__name__) class AddDatasetOwnershipConfig(TransformerSemanticsConfigModel): @@ -27,6 +34,8 @@ class AddDatasetOwnershipConfig(TransformerSemanticsConfigModel): _resolve_owner_fn = pydantic_resolve_key("get_owners_to_add") + is_container: bool = False + class AddDatasetOwnership(DatasetOwnershipTransformer): """Transformer that adds owners to datasets according to a callback function.""" @@ -70,6 +79,52 @@ def _merge_with_server_ownership( return mce_ownership + def handle_end_of_stream( + self, + ) -> List[Union[MetadataChangeProposalWrapper, MetadataChangeProposalClass]]: + if not self.config.is_container: + return [] + + logger.debug("Generating Ownership for containers") + ownership_container_mapping: Dict[str, List[OwnerClass]] = {} + for entity_urn, data_ownerships in ( + (urn, self.config.get_owners_to_add(urn)) for urn in self.entity_map.keys() + ): + if not data_ownerships: + continue + + assert self.ctx.graph + browse_paths = self.ctx.graph.get_aspect(entity_urn, BrowsePathsV2Class) + if not browse_paths: + continue + + for path in browse_paths.path: + container_urn = path.urn + + if not container_urn or not container_urn.startswith( + "urn:li:container:" + ): + continue + + if container_urn not in ownership_container_mapping: + ownership_container_mapping[container_urn] = data_ownerships + else: + ownership_container_mapping[container_urn] = list( + ownership_container_mapping[container_urn] + data_ownerships + ) + + mcps: List[ + Union[MetadataChangeProposalWrapper, MetadataChangeProposalClass] + ] = [] + + for urn, owners in ownership_container_mapping.items(): + patch_builder = DashboardPatchBuilder(urn) + for owner in owners: + patch_builder.add_owner(owner) + mcps.extend(list(patch_builder.build())) + + return mcps + def transform_aspect( self, entity_urn: str, aspect_name: str, aspect: Optional[Aspect] ) -> Optional[Aspect]: @@ -147,6 +202,7 @@ def create( class PatternDatasetOwnershipConfig(DatasetOwnershipBaseConfig): owner_pattern: KeyValuePattern = KeyValuePattern.all() default_actor: str = builder.make_user_urn("etl") + is_container: bool = False class PatternAddDatasetOwnership(AddDatasetOwnership): @@ -169,6 +225,7 @@ def __init__(self, config: PatternDatasetOwnershipConfig, ctx: PipelineContext): default_actor=config.default_actor, semantics=config.semantics, replace_existing=config.replace_existing, + is_container=config.is_container, ) super().__init__(generic_config, ctx) diff --git a/metadata-ingestion/src/datahub/ingestion/transformer/dataset_domain.py b/metadata-ingestion/src/datahub/ingestion/transformer/dataset_domain.py index 82dd21bbdd1d1..6a83824815265 100644 --- a/metadata-ingestion/src/datahub/ingestion/transformer/dataset_domain.py +++ b/metadata-ingestion/src/datahub/ingestion/transformer/dataset_domain.py @@ -1,4 +1,5 @@ -from typing import Callable, List, Optional, Union, cast +import logging +from typing import Callable, Dict, List, Optional, Sequence, Union, cast from datahub.configuration.common import ( ConfigurationError, @@ -8,12 +9,19 @@ ) from datahub.configuration.import_resolver import pydantic_resolve_key from datahub.emitter.mce_builder import Aspect +from datahub.emitter.mcp import MetadataChangeProposalWrapper from datahub.ingestion.api.common import PipelineContext from datahub.ingestion.graph.client import DataHubGraph from datahub.ingestion.transformer.dataset_transformer import DatasetDomainTransformer -from datahub.metadata.schema_classes import DomainsClass +from datahub.metadata.schema_classes import ( + BrowsePathsV2Class, + DomainsClass, + MetadataChangeProposalClass, +) from datahub.utilities.registries.domain_registry import DomainRegistry +logger = logging.getLogger(__name__) + class AddDatasetDomainSemanticsConfig(TransformerSemanticsConfigModel): get_domains_to_add: Union[ @@ -23,6 +31,8 @@ class AddDatasetDomainSemanticsConfig(TransformerSemanticsConfigModel): _resolve_domain_fn = pydantic_resolve_key("get_domains_to_add") + is_container: bool = False + class SimpleDatasetDomainSemanticsConfig(TransformerSemanticsConfigModel): domains: List[str] @@ -30,6 +40,7 @@ class SimpleDatasetDomainSemanticsConfig(TransformerSemanticsConfigModel): class PatternDatasetDomainSemanticsConfig(TransformerSemanticsConfigModel): domain_pattern: KeyValuePattern = KeyValuePattern.all() + is_container: bool = False class AddDatasetDomain(DatasetDomainTransformer): @@ -90,6 +101,56 @@ def _merge_with_server_domains( return mce_domain + def handle_end_of_stream( + self, + ) -> Sequence[Union[MetadataChangeProposalWrapper, MetadataChangeProposalClass]]: + domain_mcps: List[MetadataChangeProposalWrapper] = [] + container_domain_mapping: Dict[str, List[str]] = {} + + logger.debug("Generating Domains for containers") + + if not self.config.is_container: + return domain_mcps + + for entity_urn, domain_to_add in ( + (urn, self.config.get_domains_to_add(urn)) for urn in self.entity_map.keys() + ): + if not domain_to_add or not domain_to_add.domains: + continue + + assert self.ctx.graph + browse_paths = self.ctx.graph.get_aspect(entity_urn, BrowsePathsV2Class) + if not browse_paths: + continue + + for path in browse_paths.path: + container_urn = path.urn + + if not container_urn or not container_urn.startswith( + "urn:li:container:" + ): + continue + + if container_urn not in container_domain_mapping: + container_domain_mapping[container_urn] = domain_to_add.domains + else: + container_domain_mapping[container_urn] = list( + set( + container_domain_mapping[container_urn] + + domain_to_add.domains + ) + ) + + for urn, domains in container_domain_mapping.items(): + domain_mcps.append( + MetadataChangeProposalWrapper( + entityUrn=urn, + aspect=DomainsClass(domains=domains), + ) + ) + + return domain_mcps + def transform_aspect( self, entity_urn: str, aspect_name: str, aspect: Optional[Aspect] ) -> Optional[Aspect]: @@ -156,6 +217,7 @@ def resolve_domain(domain_urn: str) -> DomainsClass: get_domains_to_add=resolve_domain, semantics=config.semantics, replace_existing=config.replace_existing, + is_container=config.is_container, ) super().__init__(generic_config, ctx) diff --git a/metadata-ingestion/tests/unit/test_transform_dataset.py b/metadata-ingestion/tests/unit/test_transform_dataset.py index 506bfd9c12674..46c6390b184d3 100644 --- a/metadata-ingestion/tests/unit/test_transform_dataset.py +++ b/metadata-ingestion/tests/unit/test_transform_dataset.py @@ -1105,6 +1105,354 @@ def test_pattern_dataset_ownership_with_invalid_type_transformation(mock_time): ) +def test_pattern_container_and_dataset_ownership_transformation( + mock_time, mock_datahub_graph +): + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> Optional[models.BrowsePathsV2Class]: + return models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ), + models.BrowsePathEntryClass( + id="container_2", urn="urn:li:container:container_2" + ), + ] + ) + + pipeline_context = PipelineContext( + run_id="test_pattern_container_and_dataset_ownership_transformation" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig()) + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + # No owner aspect for the first dataset + no_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example1,PROD)", + aspects=[models.StatusClass(removed=False)], + ), + ) + # Dataset with an existing owner + with_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example2,PROD)", + aspects=[ + models.OwnershipClass( + owners=[ + models.OwnerClass( + owner=builder.make_user_urn("fake_owner"), + type=models.OwnershipTypeClass.DATAOWNER, + ), + ], + lastModified=models.AuditStampClass( + time=1625266033123, actor="urn:li:corpuser:datahub" + ), + ) + ], + ), + ) + + # Not a dataset, should be ignored + not_a_dataset = models.MetadataChangeEventClass( + proposedSnapshot=models.DataJobSnapshotClass( + urn="urn:li:dataJob:(urn:li:dataFlow:(airflow,dag_abc,PROD),task_456)", + aspects=[ + models.DataJobInfoClass( + name="User Deletions", + description="Constructs the fct_users_deleted from logging_events", + type=models.AzkabanJobTypeClass.SQL, + ) + ], + ) + ) + + inputs = [ + no_owner_aspect, + with_owner_aspect, + not_a_dataset, + EndOfStream(), + ] + + # Initialize the transformer with container support + transformer = PatternAddDatasetOwnership.create( + { + "owner_pattern": { + "rules": { + ".*example1.*": [builder.make_user_urn("person1")], + ".*example2.*": [builder.make_user_urn("person2")], + } + }, + "ownership_type": "DATAOWNER", + "is_container": True, # Enable container ownership handling + }, + pipeline_context, + ) + + outputs = list( + transformer.transform([RecordEnvelope(input, metadata={}) for input in inputs]) + ) + + assert len(outputs) == len(inputs) + 3 + + # Check the first entry. + assert inputs[0] == outputs[0].record + + # Check the ownership for the first dataset (example1) + first_ownership_aspect = outputs[3].record.aspect + assert first_ownership_aspect + assert len(first_ownership_aspect.owners) == 1 + assert all( + [ + owner.type == models.OwnershipTypeClass.DATAOWNER + for owner in first_ownership_aspect.owners + ] + ) + + # Check the ownership for the second dataset (example2) + second_ownership_aspect = builder.get_aspect_if_available( + outputs[1].record, models.OwnershipClass + ) + assert second_ownership_aspect + assert len(second_ownership_aspect.owners) == 2 # One existing + one new + assert all( + [ + owner.type == models.OwnershipTypeClass.DATAOWNER + for owner in second_ownership_aspect.owners + ] + ) + + # Check container ownerships + for i in range(2): + container_ownership_aspect = outputs[i + 4].record.aspect + assert container_ownership_aspect + ownership = json.loads(container_ownership_aspect.value.decode("utf-8")) + assert len(ownership) == 2 + assert ownership[0]["value"]["owner"] == builder.make_user_urn("person1") + assert ownership[1]["value"]["owner"] == builder.make_user_urn("person2") + + # Verify that the third input (not a dataset) is unchanged + assert inputs[2] == outputs[2].record + + +def test_pattern_container_and_dataset_ownership_with_no_container( + mock_time, mock_datahub_graph +): + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> Optional[models.BrowsePathsV2Class]: + return None + + pipeline_context = PipelineContext( + run_id="test_pattern_container_and_dataset_ownership_with_no_container" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig()) + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + # No owner aspect for the first dataset + no_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example1,PROD)", + aspects=[ + models.StatusClass(removed=False), + models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ), + models.BrowsePathEntryClass( + id="container_2", urn="urn:li:container:container_2" + ), + ] + ), + ], + ), + ) + # Dataset with an existing owner + with_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example2,PROD)", + aspects=[ + models.OwnershipClass( + owners=[ + models.OwnerClass( + owner=builder.make_user_urn("fake_owner"), + type=models.OwnershipTypeClass.DATAOWNER, + ), + ], + lastModified=models.AuditStampClass( + time=1625266033123, actor="urn:li:corpuser:datahub" + ), + ), + models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ), + models.BrowsePathEntryClass( + id="container_2", urn="urn:li:container:container_2" + ), + ] + ), + ], + ), + ) + + inputs = [ + no_owner_aspect, + with_owner_aspect, + EndOfStream(), + ] + + # Initialize the transformer with container support + transformer = PatternAddDatasetOwnership.create( + { + "owner_pattern": { + "rules": { + ".*example1.*": [builder.make_user_urn("person1")], + ".*example2.*": [builder.make_user_urn("person2")], + } + }, + "ownership_type": "DATAOWNER", + "is_container": True, # Enable container ownership handling + }, + pipeline_context, + ) + + outputs = list( + transformer.transform([RecordEnvelope(input, metadata={}) for input in inputs]) + ) + + assert len(outputs) == len(inputs) + 1 + + # Check the ownership for the first dataset (example1) + first_ownership_aspect = outputs[2].record.aspect + assert first_ownership_aspect + assert len(first_ownership_aspect.owners) == 1 + assert all( + [ + owner.type == models.OwnershipTypeClass.DATAOWNER + for owner in first_ownership_aspect.owners + ] + ) + + # Check the ownership for the second dataset (example2) + second_ownership_aspect = builder.get_aspect_if_available( + outputs[1].record, models.OwnershipClass + ) + assert second_ownership_aspect + assert len(second_ownership_aspect.owners) == 2 # One existing + one new + assert all( + [ + owner.type == models.OwnershipTypeClass.DATAOWNER + for owner in second_ownership_aspect.owners + ] + ) + + +def test_pattern_container_and_dataset_ownership_with_no_match( + mock_time, mock_datahub_graph +): + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> models.BrowsePathsV2Class: + return models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ) + ] + ) + + pipeline_context = PipelineContext( + run_id="test_pattern_container_and_dataset_ownership_with_no_match" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig()) + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + # No owner aspect for the first dataset + no_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example1,PROD)", + aspects=[ + models.StatusClass(removed=False), + ], + ), + ) + # Dataset with an existing owner + with_owner_aspect = models.MetadataChangeEventClass( + proposedSnapshot=models.DatasetSnapshotClass( + urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example2,PROD)", + aspects=[ + models.OwnershipClass( + owners=[ + models.OwnerClass( + owner=builder.make_user_urn("fake_owner"), + type=models.OwnershipTypeClass.DATAOWNER, + ), + ], + lastModified=models.AuditStampClass( + time=1625266033123, actor="urn:li:corpuser:datahub" + ), + ) + ], + ), + ) + + inputs = [ + no_owner_aspect, + with_owner_aspect, + EndOfStream(), + ] + + # Initialize the transformer with container support + transformer = PatternAddDatasetOwnership.create( + { + "owner_pattern": { + "rules": { + ".*example3.*": [builder.make_user_urn("person1")], + ".*example4.*": [builder.make_user_urn("person2")], + } + }, + "ownership_type": "DATAOWNER", + "is_container": True, # Enable container ownership handling + }, + pipeline_context, + ) + + outputs = list( + transformer.transform([RecordEnvelope(input, metadata={}) for input in inputs]) + ) + + assert len(outputs) == len(inputs) + 1 + + # Check the ownership for the first dataset (example1) + first_ownership_aspect = outputs[2].record.aspect + assert first_ownership_aspect + assert builder.make_user_urn("person1") not in first_ownership_aspect.owners + assert builder.make_user_urn("person2") not in first_ownership_aspect.owners + + # Check the ownership for the second dataset (example2) + second_ownership_aspect = builder.get_aspect_if_available( + outputs[1].record, models.OwnershipClass + ) + assert second_ownership_aspect + assert len(second_ownership_aspect.owners) == 1 + assert builder.make_user_urn("person1") not in second_ownership_aspect.owners + assert builder.make_user_urn("person2") not in second_ownership_aspect.owners + assert ( + builder.make_user_urn("fake_owner") == second_ownership_aspect.owners[0].owner + ) + + def gen_owners( owners: List[str], ownership_type: Union[ @@ -2435,6 +2783,224 @@ def fake_ownership_class(entity_urn: str) -> models.OwnershipClass: assert server_owner in owner_urns +def test_pattern_container_and_dataset_domain_transformation(mock_datahub_graph): + datahub_domain = builder.make_domain_urn("datahubproject.io") + acryl_domain = builder.make_domain_urn("acryl_domain") + server_domain = builder.make_domain_urn("server_domain") + + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> models.BrowsePathsV2Class: + return models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ), + models.BrowsePathEntryClass( + id="container_2", urn="urn:li:container:container_2" + ), + ] + ) + + pipeline_context = PipelineContext( + run_id="test_pattern_container_and_dataset_domain_transformation" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig()) + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + with_domain_aspect = make_generic_dataset_mcp( + aspect=models.DomainsClass(domains=[datahub_domain]), aspect_name="domains" + ) + no_domain_aspect = make_generic_dataset_mcp( + entity_urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example2,PROD)" + ) + + # Not a dataset, should be ignored + not_a_dataset = models.MetadataChangeEventClass( + proposedSnapshot=models.DataJobSnapshotClass( + urn="urn:li:dataJob:(urn:li:dataFlow:(airflow,dag_abc,PROD),task_456)", + aspects=[ + models.DataJobInfoClass( + name="User Deletions", + description="Constructs the fct_users_deleted from logging_events", + type=models.AzkabanJobTypeClass.SQL, + ) + ], + ) + ) + + inputs = [ + with_domain_aspect, + no_domain_aspect, + not_a_dataset, + EndOfStream(), + ] + + # Initialize the transformer with container support for domains + transformer = PatternAddDatasetDomain.create( + { + "domain_pattern": { + "rules": { + ".*example1.*": [acryl_domain, server_domain], + ".*example2.*": [server_domain], + } + }, + "is_container": True, # Enable container domain handling + }, + pipeline_context, + ) + + outputs = list( + transformer.transform([RecordEnvelope(input, metadata={}) for input in inputs]) + ) + + assert ( + len(outputs) == len(inputs) + 3 + ) # MCPs for the dataset without domains and the containers + + first_domain_aspect = outputs[0].record.aspect + assert first_domain_aspect + assert len(first_domain_aspect.domains) == 3 + assert all( + domain in first_domain_aspect.domains + for domain in [datahub_domain, acryl_domain, server_domain] + ) + + second_domain_aspect = outputs[3].record.aspect + assert second_domain_aspect + assert len(second_domain_aspect.domains) == 1 + assert server_domain in second_domain_aspect.domains + + # Verify that the third input (not a dataset) is unchanged + assert inputs[2] == outputs[2].record + + # Verify conainer 1 and container 2 should contain all domains + container_1 = outputs[4].record.aspect + assert len(container_1.domains) == 2 + assert acryl_domain in container_1.domains + assert server_domain in container_1.domains + + container_2 = outputs[5].record.aspect + assert len(container_2.domains) == 2 + assert acryl_domain in container_2.domains + assert server_domain in container_2.domains + + +def test_pattern_container_and_dataset_domain_transformation_with_no_container( + mock_datahub_graph, +): + datahub_domain = builder.make_domain_urn("datahubproject.io") + acryl_domain = builder.make_domain_urn("acryl_domain") + server_domain = builder.make_domain_urn("server_domain") + + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> Optional[models.BrowsePathsV2Class]: + return None + + pipeline_context = PipelineContext( + run_id="test_pattern_container_and_dataset_domain_transformation_with_no_container" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig()) + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + with_domain_aspect = make_generic_dataset_mcp( + aspect=models.DomainsClass(domains=[datahub_domain]), aspect_name="domains" + ) + no_domain_aspect = make_generic_dataset_mcp( + entity_urn="urn:li:dataset:(urn:li:dataPlatform:bigquery,example2,PROD)" + ) + + inputs = [ + with_domain_aspect, + no_domain_aspect, + EndOfStream(), + ] + + # Initialize the transformer with container support for domains + transformer = PatternAddDatasetDomain.create( + { + "domain_pattern": { + "rules": { + ".*example1.*": [acryl_domain, server_domain], + ".*example2.*": [server_domain], + } + }, + "is_container": True, # Enable container domain handling + }, + pipeline_context, + ) + + outputs = list( + transformer.transform([RecordEnvelope(input, metadata={}) for input in inputs]) + ) + + assert len(outputs) == len(inputs) + 1 + + first_domain_aspect = outputs[0].record.aspect + assert first_domain_aspect + assert len(first_domain_aspect.domains) == 3 + assert all( + domain in first_domain_aspect.domains + for domain in [datahub_domain, acryl_domain, server_domain] + ) + + second_domain_aspect = outputs[2].record.aspect + assert second_domain_aspect + assert len(second_domain_aspect.domains) == 1 + assert server_domain in second_domain_aspect.domains + + +def test_pattern_add_container_dataset_domain_no_match(mock_datahub_graph): + acryl_domain = builder.make_domain_urn("acryl.io") + datahub_domain = builder.make_domain_urn("datahubproject.io") + pattern = "urn:li:dataset:\\(urn:li:dataPlatform:invalid,.*" + + pipeline_context: PipelineContext = PipelineContext( + run_id="test_simple_add_dataset_domain" + ) + pipeline_context.graph = mock_datahub_graph(DatahubClientConfig) + + def fake_get_aspect( + entity_urn: str, + aspect_type: Type[models.BrowsePathsV2Class], + version: int = 0, + ) -> models.BrowsePathsV2Class: + return models.BrowsePathsV2Class( + path=[ + models.BrowsePathEntryClass( + id="container_1", urn="urn:li:container:container_1" + ) + ] + ) + + pipeline_context.graph.get_aspect = fake_get_aspect # type: ignore + + output = run_dataset_transformer_pipeline( + transformer_type=PatternAddDatasetDomain, + aspect=models.DomainsClass(domains=[datahub_domain]), + config={ + "replace_existing": True, + "domain_pattern": {"rules": {pattern: [acryl_domain]}}, + "is_container": True, + }, + pipeline_context=pipeline_context, + ) + + assert len(output) == 2 + assert output[0] is not None + assert output[0].record is not None + assert isinstance(output[0].record, MetadataChangeProposalWrapper) + assert output[0].record.aspect is not None + assert isinstance(output[0].record.aspect, models.DomainsClass) + transformed_aspect = cast(models.DomainsClass, output[0].record.aspect) + assert len(transformed_aspect.domains) == 0 + + def run_pattern_dataset_schema_terms_transformation_semantics( semantics: TransformerSemantics, mock_datahub_graph: Callable[[DatahubClientConfig], DataHubGraph], From e1514d5e8eb2cbb90b31bc9963d8de9d0d45b089 Mon Sep 17 00:00:00 2001 From: skrydal Date: Tue, 1 Oct 2024 21:51:00 +0200 Subject: [PATCH 6/9] fix(ingestion/nifi): Improve nifi lineage extraction performance (#11490) --- .../src/datahub/ingestion/source/nifi.py | 187 +++++++++++++----- .../tests/unit/test_nifi_source.py | 5 +- 2 files changed, 142 insertions(+), 50 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/nifi.py b/metadata-ingestion/src/datahub/ingestion/source/nifi.py index 52dce3a8b7599..25781cd2f1dcc 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/nifi.py +++ b/metadata-ingestion/src/datahub/ingestion/source/nifi.py @@ -2,10 +2,11 @@ import logging import ssl import time +from collections import defaultdict from dataclasses import dataclass, field from datetime import datetime, timedelta, timezone from enum import Enum -from typing import Callable, Dict, Iterable, List, Optional, Tuple, Union +from typing import Callable, Dict, Iterable, List, Optional, Set, Union from urllib.parse import urljoin import requests @@ -196,6 +197,75 @@ def validator_site_url(cls, site_url: str) -> str: return site_url +class BidirectionalComponentGraph: + def __init__(self): + self._outgoing: Dict[str, Set[str]] = defaultdict(set) + self._incoming: Dict[str, Set[str]] = defaultdict(set) + # this will not count duplicates/removal of non-existing connections correctly - it is only there for a quick check + self._connections_cnt = 0 + + def add_connection(self, from_component: str, to_component: str) -> None: + # this is sanity check + outgoing_duplicated = to_component in self._outgoing[from_component] + incoming_duplicated = from_component in self._incoming[to_component] + + self._outgoing[from_component].add(to_component) + self._incoming[to_component].add(from_component) + self._connections_cnt += 1 + + if outgoing_duplicated or incoming_duplicated: + logger.warning( + f"Somehow we attempted to add a connection between 2 components which already existed! Duplicated incoming: {incoming_duplicated}, duplicated outgoing: {outgoing_duplicated}. Connection from component: {from_component} to component: {to_component}" + ) + + def remove_connection(self, from_component: str, to_component: str) -> None: + self._outgoing[from_component].discard(to_component) + self._incoming[to_component].discard(from_component) + self._connections_cnt -= 1 + + def get_outgoing_connections(self, component: str) -> Set[str]: + return self._outgoing[component] + + def get_incoming_connections(self, component: str) -> Set[str]: + return self._incoming[component] + + def delete_component(self, component: str) -> None: + logger.debug(f"Deleting component with id: {component}") + incoming = self._incoming[component] + logger.debug( + f"Recognized {len(incoming)} incoming connections to the component" + ) + outgoing = self._outgoing[component] + logger.debug( + f"Recognized {len(outgoing)} outgoing connections from the component" + ) + + for i in incoming: + for o in outgoing: + self.add_connection(i, o) + + for i in incoming: + self._outgoing[i].remove(component) + for o in outgoing: + self._incoming[o].remove(component) + + added_connections_cnt = len(incoming) * len(outgoing) + deleted_connections_cnt = len(incoming) + len(outgoing) + logger.debug( + f"Deleted {deleted_connections_cnt} connections and added {added_connections_cnt}" + ) + + del self._outgoing[component] + del self._incoming[component] + + # for performance reasons we are not using `remove_connection` function when deleting an entire component, + # therefor we need to adjust the estimated count + self._connections_cnt -= deleted_connections_cnt + + def __len__(self): + return self._connections_cnt + + TOKEN_ENDPOINT = "access/token" KERBEROS_TOKEN_ENDPOINT = "access/kerberos" ABOUT_ENDPOINT = "flow/about" @@ -360,7 +430,9 @@ class NifiFlow: root_process_group: NifiProcessGroup components: Dict[str, NifiComponent] = field(default_factory=dict) remotely_accessible_ports: Dict[str, NifiComponent] = field(default_factory=dict) - connections: List[Tuple[str, str]] = field(default_factory=list) + connections: BidirectionalComponentGraph = field( + default_factory=BidirectionalComponentGraph + ) processGroups: Dict[str, NifiProcessGroup] = field(default_factory=dict) remoteProcessGroups: Dict[str, NifiRemoteProcessGroup] = field(default_factory=dict) remote_ports: Dict[str, NifiComponent] = field(default_factory=dict) @@ -416,10 +488,15 @@ def create(cls, config_dict: dict, ctx: PipelineContext) -> "Source": def get_report(self) -> SourceReport: return self.report - def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 + def update_flow( + self, pg_flow_dto: Dict, recursion_level: int = 0 + ) -> None: # noqa: C901 """ Update self.nifi_flow with contents of the input process group `pg_flow_dto` """ + logger.debug( + f"Updating flow with pg_flow_dto {pg_flow_dto.get('breadcrumb', {}).get('breadcrumb', {}).get('id')}, recursion level: {recursion_level}" + ) breadcrumb_dto = pg_flow_dto.get("breadcrumb", {}).get("breadcrumb", {}) nifi_pg = NifiProcessGroup( breadcrumb_dto.get("id"), @@ -433,6 +510,7 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 flow_dto = pg_flow_dto.get("flow", {}) + logger.debug(f"Processing {len(flow_dto.get('processors', []))} processors") for processor in flow_dto.get("processors", []): component = processor.get("component") self.nifi_flow.components[component.get("id")] = NifiComponent( @@ -445,6 +523,7 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 comments=component.get("config", {}).get("comments"), status=component.get("status", {}).get("runStatus"), ) + logger.debug(f"Processing {len(flow_dto.get('funnels', []))} funnels") for funnel in flow_dto.get("funnels", []): component = funnel.get("component") self.nifi_flow.components[component.get("id")] = NifiComponent( @@ -458,13 +537,15 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 ) logger.debug(f"Adding funnel {component.get('id')}") + logger.debug(f"Processing {len(flow_dto.get('connections', []))} connections") for connection in flow_dto.get("connections", []): # Exclude self - recursive relationships if connection.get("sourceId") != connection.get("destinationId"): - self.nifi_flow.connections.append( - (connection.get("sourceId"), connection.get("destinationId")) + self.nifi_flow.connections.add_connection( + connection.get("sourceId"), connection.get("destinationId") ) + logger.debug(f"Processing {len(flow_dto.get('inputPorts', []))} inputPorts") for inputPort in flow_dto.get("inputPorts", []): component = inputPort.get("component") if inputPort.get("allowRemoteAccess"): @@ -492,6 +573,7 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 ) logger.debug(f"Adding port {component.get('id')}") + logger.debug(f"Processing {len(flow_dto.get('outputPorts', []))} outputPorts") for outputPort in flow_dto.get("outputPorts", []): component = outputPort.get("component") if outputPort.get("allowRemoteAccess"): @@ -519,6 +601,9 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 ) logger.debug(f"Adding report port {component.get('id')}") + logger.debug( + f"Processing {len(flow_dto.get('remoteProcessGroups', []))} remoteProcessGroups" + ) for rpg in flow_dto.get("remoteProcessGroups", []): rpg_component = rpg.get("component", {}) remote_ports = {} @@ -564,7 +649,13 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 self.nifi_flow.components.update(remote_ports) self.nifi_flow.remoteProcessGroups[nifi_rpg.id] = nifi_rpg + logger.debug( + f"Processing {len(flow_dto.get('processGroups', []))} processGroups" + ) for pg in flow_dto.get("processGroups", []): + logger.debug( + f"Retrieving process group: {pg.get('id')} while updating flow for {pg_flow_dto.get('breadcrumb', {}).get('breadcrumb', {}).get('id')}" + ) pg_response = self.session.get( url=urljoin(self.rest_api_base_url, PG_ENDPOINT) + pg.get("id") ) @@ -578,11 +669,24 @@ def update_flow(self, pg_flow_dto: Dict) -> None: # noqa: C901 pg_flow_dto = pg_response.json().get("processGroupFlow", {}) - self.update_flow(pg_flow_dto) + self.update_flow(pg_flow_dto, recursion_level=recursion_level + 1) def update_flow_keep_only_ingress_egress(self): components_to_del: List[NifiComponent] = [] - for component in self.nifi_flow.components.values(): + components = self.nifi_flow.components.values() + logger.debug( + f"Processing {len(components)} components for keep only ingress/egress" + ) + logger.debug( + f"All the connections recognized: {len(self.nifi_flow.connections)}" + ) + for index, component in enumerate(components, start=1): + logger.debug( + f"Processing {index}th component for ingress/egress pruning. Component id: {component.id}, name: {component.name}, type: {component.type}" + ) + logger.debug( + f"Current amount of connections: {len(self.nifi_flow.connections)}" + ) if ( component.nifi_type is NifiType.PROCESSOR and component.type @@ -592,47 +696,28 @@ def update_flow_keep_only_ingress_egress(self): NifiType.REMOTE_INPUT_PORT, NifiType.REMOTE_OUTPUT_PORT, ]: + self.nifi_flow.connections.delete_component(component.id) components_to_del.append(component) - incoming = list( - filter(lambda x: x[1] == component.id, self.nifi_flow.connections) - ) - outgoing = list( - filter(lambda x: x[0] == component.id, self.nifi_flow.connections) - ) - # Create new connections from incoming to outgoing - for i in incoming: - for j in outgoing: - self.nifi_flow.connections.append((i[0], j[1])) - - # Remove older connections, as we already created - # new connections bypassing component to be deleted - - for i in incoming: - self.nifi_flow.connections.remove(i) - for j in outgoing: - self.nifi_flow.connections.remove(j) - - for c in components_to_del: - if c.nifi_type is NifiType.PROCESSOR and ( - c.name.startswith("Get") - or c.name.startswith("List") - or c.name.startswith("Fetch") - or c.name.startswith("Put") + + for component in components_to_del: + if component.nifi_type is NifiType.PROCESSOR and component.name.startswith( + ("Get", "List", "Fetch", "Put") ): self.report.warning( - f"Dropping NiFi Processor of type {c.type}, id {c.id}, name {c.name} from lineage view. \ + f"Dropping NiFi Processor of type {component.type}, id {component.id}, name {component.name} from lineage view. \ This is likely an Ingress or Egress node which may be reading to/writing from external datasets \ However not currently supported in datahub", self.config.site_url, ) else: logger.debug( - f"Dropping NiFi Component of type {c.type}, id {c.id}, name {c.name} from lineage view." + f"Dropping NiFi Component of type {component.type}, id {component.id}, name {component.name} from lineage view." ) - del self.nifi_flow.components[c.id] + del self.nifi_flow.components[component.id] def create_nifi_flow(self): + logger.debug(f"Retrieving NIFI info from {ABOUT_ENDPOINT}") about_response = self.session.get( url=urljoin(self.rest_api_base_url, ABOUT_ENDPOINT) ) @@ -646,6 +731,8 @@ def create_nifi_flow(self): ) else: logger.warning("Failed to fetch version for nifi") + logger.debug(f"Retrieved nifi version: {nifi_version}") + logger.debug(f"Retrieving cluster info from {CLUSTER_ENDPOINT}") cluster_response = self.session.get( url=urljoin(self.rest_api_base_url, CLUSTER_ENDPOINT) ) @@ -654,8 +741,10 @@ def create_nifi_flow(self): clustered = ( cluster_response.json().get("clusterSummary", {}).get("clustered") ) + logger.debug(f"Retrieved cluster summary: {clustered}") else: logger.warning("Failed to fetch cluster summary for flow") + logger.debug("Retrieving ROOT Process Group") pg_response = self.session.get( url=urljoin(self.rest_api_base_url, PG_ENDPOINT) + "root" ) @@ -695,7 +784,7 @@ def fetch_provenance_events( if provenance_response.ok: provenance = provenance_response.json().get("provenance", {}) provenance_uri = provenance.get("uri") - + logger.debug(f"Retrieving provenance uri: {provenance_uri}") provenance_response = self.session.get(provenance_uri) if provenance_response.ok: provenance = provenance_response.json().get("provenance", {}) @@ -734,7 +823,9 @@ def fetch_provenance_events( total = provenance.get("results", {}).get("total") totalCount = provenance.get("results", {}).get("totalCount") + logger.debug(f"Retrieved {totalCount} of {total}") if total != str(totalCount): + logger.debug("Trying to retrieve more events for the same processor") yield from self.fetch_provenance_events( processor, eventType, startDate, oldest_event_time ) @@ -800,6 +891,7 @@ def submit_provenance_query(self, processor, eventType, startDate, endDate): return provenance_response def delete_provenance(self, provenance_uri): + logger.debug(f"Deleting provenance with uri: {provenance_uri}") delete_response = self.session.delete(provenance_uri) if not delete_response.ok: logger.error("failed to delete provenance ", provenance_uri) @@ -821,12 +913,8 @@ def construct_workunits(self) -> Iterable[MetadataWorkUnit]: # noqa: C901 job_name = component.name job_urn = builder.make_data_job_urn_with_flow(flow_urn, component.id) - incoming = list( - filter(lambda x: x[1] == component.id, self.nifi_flow.connections) - ) - outgoing = list( - filter(lambda x: x[0] == component.id, self.nifi_flow.connections) - ) + incoming = self.nifi_flow.connections.get_incoming_connections(component.id) + outgoing = self.nifi_flow.connections.get_outgoing_connections(component.id) inputJobs = set() jobProperties = None @@ -864,8 +952,7 @@ def construct_workunits(self) -> Iterable[MetadataWorkUnit]: # noqa: C901 datasetProperties=dataset.dataset_properties, ) - for edge in incoming: - incoming_from = edge[0] + for incoming_from in incoming: if incoming_from in self.nifi_flow.remotely_accessible_ports.keys(): dataset_name = f"{self.config.site_name}.{self.nifi_flow.remotely_accessible_ports[incoming_from].name}" dataset_urn = builder.make_dataset_urn( @@ -882,8 +969,7 @@ def construct_workunits(self) -> Iterable[MetadataWorkUnit]: # noqa: C901 builder.make_data_job_urn_with_flow(flow_urn, incoming_from) ) - for edge in outgoing: - outgoing_to = edge[1] + for outgoing_to in outgoing: if outgoing_to in self.nifi_flow.remotely_accessible_ports.keys(): dataset_name = f"{self.config.site_name}.{self.nifi_flow.remotely_accessible_ports[outgoing_to].name}" dataset_urn = builder.make_dataset_urn( @@ -977,14 +1063,19 @@ def make_flow_urn(self) -> str: ) def process_provenance_events(self): + logger.debug("Starting processing of provenance events") startDate = datetime.now(timezone.utc) - timedelta( days=self.config.provenance_days ) eventAnalyzer = NifiProcessorProvenanceEventAnalyzer() eventAnalyzer.env = self.config.env - - for component in self.nifi_flow.components.values(): + components = self.nifi_flow.components.values() + logger.debug(f"Processing {len(components)} components") + for component in components: + logger.debug( + f"Processing provenance events for component id: {component.id} name: {component.name}" + ) if component.nifi_type is NifiType.PROCESSOR: eventType = eventAnalyzer.KNOWN_INGRESS_EGRESS_PROCESORS[component.type] events = self.fetch_provenance_events(component, eventType, startDate) diff --git a/metadata-ingestion/tests/unit/test_nifi_source.py b/metadata-ingestion/tests/unit/test_nifi_source.py index 9e8bf64261ffa..30a0855d44f34 100644 --- a/metadata-ingestion/tests/unit/test_nifi_source.py +++ b/metadata-ingestion/tests/unit/test_nifi_source.py @@ -6,6 +6,7 @@ from datahub.ingestion.api.common import PipelineContext from datahub.ingestion.source.nifi import ( + BidirectionalComponentGraph, NifiComponent, NifiFlow, NifiProcessGroup, @@ -55,7 +56,7 @@ def test_nifi_s3_provenance_event(): ) }, remotely_accessible_ports={}, - connections=[], + connections=BidirectionalComponentGraph(), processGroups={ "803ebb92-017d-1000-2961-4bdaa27a3ba0": NifiProcessGroup( id="803ebb92-017d-1000-2961-4bdaa27a3ba0", @@ -126,7 +127,7 @@ def test_nifi_s3_provenance_event(): ) }, remotely_accessible_ports={}, - connections=[], + connections=BidirectionalComponentGraph(), processGroups={ "803ebb92-017d-1000-2961-4bdaa27a3ba0": NifiProcessGroup( id="803ebb92-017d-1000-2961-4bdaa27a3ba0", From a87c1236112556d5ab91bfaba54679d06495af26 Mon Sep 17 00:00:00 2001 From: David Schmidt Date: Wed, 2 Oct 2024 05:59:23 +0200 Subject: [PATCH 7/9] fix(dagster-plugin): Fix in/outs format and source config (#11481) --- .../dagster-plugin/examples/advanced_ops_jobs.py | 5 ++--- .../dagster-plugin/examples/assets_job.py | 13 +++++-------- .../dagster-plugin/examples/ops_job.py | 15 ++++++--------- 3 files changed, 13 insertions(+), 20 deletions(-) diff --git a/metadata-ingestion-modules/dagster-plugin/examples/advanced_ops_jobs.py b/metadata-ingestion-modules/dagster-plugin/examples/advanced_ops_jobs.py index d4cc65297e42c..7b7616b1ec11d 100644 --- a/metadata-ingestion-modules/dagster-plugin/examples/advanced_ops_jobs.py +++ b/metadata-ingestion-modules/dagster-plugin/examples/advanced_ops_jobs.py @@ -32,12 +32,12 @@ def extract(): ins={ "data": In( dagster_type=PythonObjectDagsterType(list), - metadata={"datahub.inputs": [DatasetUrn("snowflake", "tableA").urn]}, + metadata={"datahub.inputs": [DatasetUrn("snowflake", "tableA").urn()]}, ) }, out={ "result": Out( - metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableB").urn]} + metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableB").urn()]} ) }, ) @@ -101,6 +101,5 @@ def asset_lineage_extractor( dagster_url="http://localhost:3000", asset_lineage_extractor=asset_lineage_extractor, ) - datahub_sensor = make_datahub_sensor(config=config) defs = Definitions(jobs=[do_stuff], sensors=[datahub_sensor]) diff --git a/metadata-ingestion-modules/dagster-plugin/examples/assets_job.py b/metadata-ingestion-modules/dagster-plugin/examples/assets_job.py index 57634ab345a5e..1ed3f2f915061 100644 --- a/metadata-ingestion-modules/dagster-plugin/examples/assets_job.py +++ b/metadata-ingestion-modules/dagster-plugin/examples/assets_job.py @@ -7,6 +7,7 @@ define_asset_job, multi_asset, ) +from datahub.ingestion.graph.config import DatahubClientConfig from datahub.utilities.urns.dataset_urn import DatasetUrn from datahub_dagster_plugin.sensors.datahub_sensors import ( @@ -18,7 +19,7 @@ @multi_asset( outs={ "extract": AssetOut( - metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableD").urn]} + metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableD").urn()]} ), } ) @@ -47,13 +48,9 @@ def transform(extract): assets_job = define_asset_job(name="assets_job") -config = DatahubDagsterSourceConfig.parse_obj( - { - "rest_sink_config": { - "server": "http://localhost:8080", - }, - "dagster_url": "http://localhost:3000", - } +config = DatahubDagsterSourceConfig( + datahub_client_config=DatahubClientConfig(server="http://localhost:8080"), + dagster_url="http://localhost:3000", ) datahub_sensor = make_datahub_sensor(config=config) diff --git a/metadata-ingestion-modules/dagster-plugin/examples/ops_job.py b/metadata-ingestion-modules/dagster-plugin/examples/ops_job.py index d743e19a235d5..a17fc89e6922d 100644 --- a/metadata-ingestion-modules/dagster-plugin/examples/ops_job.py +++ b/metadata-ingestion-modules/dagster-plugin/examples/ops_job.py @@ -1,4 +1,5 @@ from dagster import Definitions, In, Out, PythonObjectDagsterType, job, op +from datahub.ingestion.graph.config import DatahubClientConfig from datahub.utilities.urns.dataset_urn import DatasetUrn from datahub_dagster_plugin.sensors.datahub_sensors import ( @@ -17,12 +18,12 @@ def extract(): ins={ "data": In( dagster_type=PythonObjectDagsterType(list), - metadata={"datahub.inputs": [DatasetUrn("snowflake", "tableA").urn]}, + metadata={"datahub.inputs": [DatasetUrn("snowflake", "tableA").urn()]}, ) }, out={ "result": Out( - metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableB").urn]} + metadata={"datahub.outputs": [DatasetUrn("snowflake", "tableB").urn()]} ) }, ) @@ -38,13 +39,9 @@ def do_stuff(): transform(extract()) -config = DatahubDagsterSourceConfig.parse_obj( - { - "rest_sink_config": { - "server": "http://localhost:8080", - }, - "dagster_url": "http://localhost:3000", - } +config = DatahubDagsterSourceConfig( + datahub_client_config=DatahubClientConfig(server="http://localhost:8080"), + dagster_url="http://localhost:3000", ) datahub_sensor = make_datahub_sensor(config=config) From 44645b9650ea9b79d4493b9ddca3502654657bd0 Mon Sep 17 00:00:00 2001 From: Bumyu Date: Wed, 2 Oct 2024 05:59:37 +0200 Subject: [PATCH 8/9] fix(ingest/elasticsearch): detect sub-properties in 'nested' type mapping (#11338) Co-authored-by: Lawrence De Spiegeleire --- .../ingestion/source/elastic_search.py | 20 +++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/elastic_search.py b/metadata-ingestion/src/datahub/ingestion/source/elastic_search.py index 8bda5db9a379a..aa5913f5dc66b 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/elastic_search.py +++ b/metadata-ingestion/src/datahub/ingestion/source/elastic_search.py @@ -138,31 +138,31 @@ def _get_schema_fields( for columnName, column in elastic_schema_dict.items(): elastic_type: Optional[str] = column.get("type") nested_props: Optional[Dict[str, Any]] = column.get(PROPERTIES) - if elastic_type is not None: - self._prefix_name_stack.append(f"[type={elastic_type}].{columnName}") - schema_field_data_type = self.get_column_type(elastic_type) + if nested_props: + self._prefix_name_stack.append(f"[type={PROPERTIES}].{columnName}") schema_field = SchemaField( fieldPath=self._get_cur_field_path(), - nativeDataType=elastic_type, - type=schema_field_data_type, + nativeDataType=PROPERTIES, + type=SchemaFieldDataTypeClass(RecordTypeClass()), description=None, nullable=True, recursive=False, ) yield schema_field + yield from self._get_schema_fields(nested_props) self._prefix_name_stack.pop() - elif nested_props: - self._prefix_name_stack.append(f"[type={PROPERTIES}].{columnName}") + elif elastic_type is not None: + self._prefix_name_stack.append(f"[type={elastic_type}].{columnName}") + schema_field_data_type = self.get_column_type(elastic_type) schema_field = SchemaField( fieldPath=self._get_cur_field_path(), - nativeDataType=PROPERTIES, - type=SchemaFieldDataTypeClass(RecordTypeClass()), + nativeDataType=elastic_type, + type=schema_field_data_type, description=None, nullable=True, recursive=False, ) yield schema_field - yield from self._get_schema_fields(nested_props) self._prefix_name_stack.pop() else: # Unexpected! Log a warning. From fa67e3a119e0c8a2970fb0c44e1fe0cc5129ee46 Mon Sep 17 00:00:00 2001 From: Harshal Sheth Date: Wed, 2 Oct 2024 20:02:10 -0700 Subject: [PATCH 9/9] chore: update case studies (#11520) --- docs-website/adoptionStoriesIndexes.json | 4 +--- .../_components/CaseStudy/caseStudyContent.js | 20 +++++++++--------- docs-website/src/pages/index.js | 2 +- .../static/img/logos/companies/visa.png | Bin 20549 -> 0 bytes 4 files changed, 12 insertions(+), 14 deletions(-) delete mode 100644 docs-website/static/img/logos/companies/visa.png diff --git a/docs-website/adoptionStoriesIndexes.json b/docs-website/adoptionStoriesIndexes.json index 1177d74dc6df9..15cb770697c1c 100644 --- a/docs-website/adoptionStoriesIndexes.json +++ b/docs-website/adoptionStoriesIndexes.json @@ -14,8 +14,6 @@ { "name": "Visa", "slug": "visa", - "imageUrl": "/img/logos/companies/visa.png", - "imageSize": "large", "link": "https://blog.datahubproject.io/how-visa-uses-datahub-to-scale-data-governance-cace052d61c5", "linkType": "blog", "tagline": "How Visa uses DataHub to scale data governance", @@ -374,4 +372,4 @@ "category": "And More" } ] -} \ No newline at end of file +} diff --git a/docs-website/src/pages/_components/CaseStudy/caseStudyContent.js b/docs-website/src/pages/_components/CaseStudy/caseStudyContent.js index 47c379027da81..0346acc37ed4a 100644 --- a/docs-website/src/pages/_components/CaseStudy/caseStudyContent.js +++ b/docs-website/src/pages/_components/CaseStudy/caseStudyContent.js @@ -9,16 +9,16 @@ const caseStudyData = [ image: "https://datahubproject.io/img/logos/companies/netflix.png", link: "https://datahubproject.io/adoption-stories/#netflix", }, - { - title: "Scaling Data Governance", - description: - "How VISA Uses DataHub to Scale Data Governance.", - tag: "Finance", - backgroundImage: - "https://miro.medium.com/v2/resize:fit:2000/format:webp/1*mtC4j8J-jumJKWg8RuR6xQ@2x.png", - image: "https://datahubproject.io/img/logos/companies/visa.png", - link: "https://datahubproject.io/adoption-stories/#visa", - }, + // { + // title: "Scaling Data Governance", + // description: + // "How VISA Uses DataHub to Scale Data Governance.", + // tag: "Finance", + // backgroundImage: + // "https://miro.medium.com/v2/resize:fit:2000/format:webp/1*mtC4j8J-jumJKWg8RuR6xQ@2x.png", + // image: "https://datahubproject.io/img/logos/companies/visa.png", + // link: "https://datahubproject.io/adoption-stories/#visa", + // }, { title: "Ensuring Data Reliability", description: diff --git a/docs-website/src/pages/index.js b/docs-website/src/pages/index.js index d538831ca3dca..d74e80f381097 100644 --- a/docs-website/src/pages/index.js +++ b/docs-website/src/pages/index.js @@ -21,7 +21,7 @@ import CloseButton from "@ant-design/icons/CloseCircleFilled"; const companyIndexes = require("../../adoptionStoriesIndexes.json"); const companies = companyIndexes.companies; -const keyCompanySlugs = ["netflix", "visa", "pinterest", "airtel", "optum"]; +const keyCompanySlugs = ["netflix", "pinterest", "airtel", "notion", "optum"]; const keyCompanies = keyCompanySlugs .map((slug) => companies.find((co) => co.slug === slug)) .filter((isDefined) => isDefined); diff --git a/docs-website/static/img/logos/companies/visa.png b/docs-website/static/img/logos/companies/visa.png deleted file mode 100644 index 20af2c5cd8e26f5236a0945106fd31ef270aed15..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 20549 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