diff --git a/.github/workflows/publish-datahub-jars.yml b/.github/workflows/publish-datahub-jars.yml index 46421c8e5b8eba..fc9296235c3ac7 100644 --- a/.github/workflows/publish-datahub-jars.yml +++ b/.github/workflows/publish-datahub-jars.yml @@ -167,3 +167,29 @@ jobs: echo signingKey=$SIGNING_KEY >> gradle.properties ./gradlew -PreleaseVersion=${{ needs.setup.outputs.tag }} :metadata-auth:auth-api:publish ./gradlew :metadata-auth:auth-api:closeAndReleaseRepository --info + - name: publish datahub-custom-plugin-lib snapshot jar + if: ${{ github.event_name != 'release' }} + env: + RELEASE_USERNAME: ${{ secrets.RELEASE_USERNAME }} + RELEASE_PASSWORD: ${{ secrets.RELEASE_PASSWORD }} + SIGNING_PASSWORD: ${{ secrets.SIGNING_PASSWORD }} + SIGNING_KEY: ${{ secrets.SIGNING_KEY }} + NEXUS_USERNAME: ${{ secrets.NEXUS_USERNAME }} + NEXUS_PASSWORD: ${{ secrets.NEXUS_PASSWORD }} + run: | + echo signingKey=$SIGNING_KEY >> gradle.properties + ./gradlew :metadata-integration:java:custom-plugin-lib:printVersion + ./gradlew :metadata-integration:java:custom-plugin-lib:publish + - name: release datahub-custom-plugin-lib jar + if: ${{ github.event_name == 'release' }} + env: + RELEASE_USERNAME: ${{ secrets.RELEASE_USERNAME }} + RELEASE_PASSWORD: ${{ secrets.RELEASE_PASSWORD }} + SIGNING_PASSWORD: ${{ secrets.SIGNING_PASSWORD }} + SIGNING_KEY: ${{ secrets.SIGNING_KEY }} + NEXUS_USERNAME: ${{ secrets.NEXUS_USERNAME }} + NEXUS_PASSWORD: ${{ secrets.NEXUS_PASSWORD }} + run: | + echo signingKey=$SIGNING_KEY >> gradle.properties + ./gradlew -PreleaseVersion=${{ needs.setup.outputs.tag }} :metadata-integration:java:custom-plugin-lib:publish + ./gradlew :metadata-integration:java:custom-plugin-lib:closeAndReleaseRepository --info diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/Constants.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/Constants.java index 6fc6edc66f3572..f70c46ba943a5a 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/Constants.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/Constants.java @@ -1,5 +1,8 @@ package com.linkedin.datahub.graphql; +import com.google.common.collect.ImmutableSet; +import java.util.Set; + /** Constants relating to GraphQL type system & execution. */ public class Constants { @@ -28,4 +31,11 @@ private Constants() {} public static final String BROWSE_PATH_V2_DELIMITER = "␟"; public static final String VERSION_STAMP_FIELD_NAME = "versionStamp"; public static final String ENTITY_FILTER_NAME = "_entityType"; + + public static final Set DEFAULT_PERSONA_URNS = + ImmutableSet.of( + "urn:li:dataHubPersona:technicalUser", + "urn:li:dataHubPersona:businessUser", + "urn:li:dataHubPersona:dataLeader", + "urn:li:dataHubPersona:dataSteward"); } diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/GmsGraphQLEngine.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/GmsGraphQLEngine.java index b17e4bd386bdac..6f2e250c17c34e 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/GmsGraphQLEngine.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/GmsGraphQLEngine.java @@ -42,6 +42,7 @@ import com.linkedin.datahub.graphql.generated.CorpGroup; import com.linkedin.datahub.graphql.generated.CorpGroupInfo; import com.linkedin.datahub.graphql.generated.CorpUser; +import com.linkedin.datahub.graphql.generated.CorpUserEditableProperties; import com.linkedin.datahub.graphql.generated.CorpUserInfo; import com.linkedin.datahub.graphql.generated.CorpUserViewsSettings; import com.linkedin.datahub.graphql.generated.Dashboard; @@ -53,6 +54,7 @@ import com.linkedin.datahub.graphql.generated.DataHubView; import com.linkedin.datahub.graphql.generated.DataJob; import com.linkedin.datahub.graphql.generated.DataJobInputOutput; +import com.linkedin.datahub.graphql.generated.DataPlatform; import com.linkedin.datahub.graphql.generated.DataPlatformInstance; import com.linkedin.datahub.graphql.generated.DataQualityContract; import com.linkedin.datahub.graphql.generated.Dataset; @@ -1823,6 +1825,18 @@ private void configureCorpUserResolvers(final RuntimeWiring.Builder builder) { new LoadableTypeResolver<>( corpUserType, (env) -> ((CorpUserInfo) env.getSource()).getManager().getUrn()))); + builder.type( + "CorpUserEditableProperties", + typeWiring -> + typeWiring.dataFetcher( + "platforms", + new LoadableTypeBatchResolver<>( + dataPlatformType, + (env) -> + ((CorpUserEditableProperties) env.getSource()) + .getPlatforms().stream() + .map(DataPlatform::getUrn) + .collect(Collectors.toList())))); } /** diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolver.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolver.java index b1d2b40933d7e0..b720aa11a8bdc7 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolver.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolver.java @@ -72,9 +72,10 @@ public CompletableFuture get(DataFetchingEnvironment environment) throw context.getOperationContext(), assertionUrn, asserteeUrn, - input.getTimestampMillis(), - assertionResult, - mapContextParameters(input.getProperties())); + input.getTimestampMillis() != null + ? input.getTimestampMillis() + : System.currentTimeMillis(), + assertionResult); return true; } throw new AuthorizationException( @@ -99,6 +100,9 @@ private AssertionResult mapAssertionResult(AssertionResultInput input) { if (assertionResult.getType() == AssertionResultType.ERROR && input.getError() != null) { assertionResult.setError(mapAssertionResultError(input)); } + if (input.getProperties() != null) { + assertionResult.setNativeResults(mapContextParameters(input.getProperties())); + } return assertionResult; } diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/CorpUserType.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/CorpUserType.java index b1ce42e72482a5..3c2bfd7225edf5 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/CorpUserType.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/CorpUserType.java @@ -1,11 +1,13 @@ package com.linkedin.datahub.graphql.types.corpuser; +import static com.linkedin.datahub.graphql.Constants.DEFAULT_PERSONA_URNS; import static com.linkedin.datahub.graphql.resolvers.mutate.MutationUtils.*; import static com.linkedin.metadata.Constants.*; import com.datahub.authorization.ConjunctivePrivilegeGroup; import com.datahub.authorization.DisjunctivePrivilegeGroup; import com.google.common.collect.ImmutableList; +import com.linkedin.common.UrnArray; import com.linkedin.common.url.Url; import com.linkedin.common.urn.Urn; import com.linkedin.common.urn.UrnUtils; @@ -14,6 +16,8 @@ import com.linkedin.datahub.graphql.QueryContext; import com.linkedin.datahub.graphql.authorization.AuthorizationUtils; import com.linkedin.datahub.graphql.exception.AuthorizationException; +import com.linkedin.datahub.graphql.exception.DataHubGraphQLErrorCode; +import com.linkedin.datahub.graphql.exception.DataHubGraphQLException; import com.linkedin.datahub.graphql.featureflags.FeatureFlags; import com.linkedin.datahub.graphql.generated.AutoCompleteResults; import com.linkedin.datahub.graphql.generated.CorpUser; @@ -246,7 +250,20 @@ private RecordTemplate mapCorpUserEditableInfo( if (input.getEmail() != null) { result.setEmail(input.getEmail()); } - + if (input.getPlatformUrns() != null) { + result.setPlatforms( + new UrnArray( + input.getPlatformUrns().stream().map(UrnUtils::getUrn).collect(Collectors.toList()))); + } + if (input.getPersonaUrn() != null) { + if (DEFAULT_PERSONA_URNS.contains(input.getPersonaUrn())) { + result.setPersona(UrnUtils.getUrn(input.getPersonaUrn())); + } else { + throw new DataHubGraphQLException( + String.format("Provided persona urn %s does not exist", input.getPersonaUrn()), + DataHubGraphQLErrorCode.NOT_FOUND); + } + } return result; } } diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/mappers/CorpUserEditableInfoMapper.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/mappers/CorpUserEditableInfoMapper.java index 1ff2f069b8112c..38f3c75d7a9fa8 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/mappers/CorpUserEditableInfoMapper.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/corpuser/mappers/CorpUserEditableInfoMapper.java @@ -2,7 +2,10 @@ import com.linkedin.datahub.graphql.QueryContext; import com.linkedin.datahub.graphql.generated.CorpUserEditableProperties; +import com.linkedin.datahub.graphql.generated.DataHubPersona; +import com.linkedin.datahub.graphql.generated.DataPlatform; import com.linkedin.datahub.graphql.types.mappers.ModelMapper; +import java.util.stream.Collectors; import javax.annotation.Nonnull; import javax.annotation.Nullable; @@ -38,6 +41,22 @@ public CorpUserEditableProperties apply( if (info.hasPictureLink()) { result.setPictureLink(info.getPictureLink().toString()); } + if (info.hasPlatforms()) { + result.setPlatforms( + info.getPlatforms().stream() + .map( + urn -> { + DataPlatform platform = new DataPlatform(); + platform.setUrn(urn.toString()); + return platform; + }) + .collect(Collectors.toList())); + } + if (info.hasPersona()) { + DataHubPersona persona = new DataHubPersona(); + persona.setUrn(info.getPersona().toString()); + result.setPersona(persona); + } return result; } } diff --git a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/ermodelrelationship/CreateERModelRelationshipResolver.java b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/ermodelrelationship/CreateERModelRelationshipResolver.java index 61896ed1a0659f..cafd0b5ab082b2 100644 --- a/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/ermodelrelationship/CreateERModelRelationshipResolver.java +++ b/datahub-graphql-core/src/main/java/com/linkedin/datahub/graphql/types/ermodelrelationship/CreateERModelRelationshipResolver.java @@ -54,14 +54,15 @@ public CompletableFuture get(DataFetchingEnvironment enviro highDataset = source; } // The following sequence mimics datahub.emitter.mce_builder.datahub_guid + // Keys have to be in alphabetical order - Destination, ERModelRelationName and Source String ermodelrelationKey = - "{\"Source\":\"" + "{\"Destination\":\"" + lowDataset - + "\",\"Destination\":\"" - + highDataset + "\",\"ERModelRelationName\":\"" + ermodelrelationName + + "\",\"Source\":\"" + + highDataset + "\"}"; byte[] mybytes = ermodelrelationKey.getBytes(StandardCharsets.UTF_8); diff --git a/datahub-graphql-core/src/main/resources/assertions.graphql b/datahub-graphql-core/src/main/resources/assertions.graphql index be9cf61069635e..ff182089ad7ff6 100644 --- a/datahub-graphql-core/src/main/resources/assertions.graphql +++ b/datahub-graphql-core/src/main/resources/assertions.graphql @@ -87,7 +87,9 @@ input AssertionResultInput { type: AssertionResultType! """ - Additional key-value pairs representing runtime context + Additional metadata representing about the native results of the assertion. + These will be displayed alongside the result. + It should be used to capture additional context that is useful for the user. """ properties: [StringMapEntryInput!] diff --git a/datahub-graphql-core/src/main/resources/entity.graphql b/datahub-graphql-core/src/main/resources/entity.graphql index d48a9976e15d77..89c7b4a4cd0556 100644 --- a/datahub-graphql-core/src/main/resources/entity.graphql +++ b/datahub-graphql-core/src/main/resources/entity.graphql @@ -4139,6 +4139,16 @@ type CorpUserEditableProperties { Email address for the user """ email: String + + """ + User persona, if present + """ + persona: DataHubPersona + + """ + Platforms commonly used by the user, if present. + """ + platforms: [DataPlatform!] } """ @@ -4189,6 +4199,16 @@ input CorpUserUpdateInput { Email address for the user """ email: String + + """ + The platforms that the user frequently works with + """ + platformUrns: [String!] + + """ + The user's persona urn" + """ + personaUrn: String } """ @@ -12142,6 +12162,7 @@ input CreateDataProductPropertiesInput { description: String } + """ Input properties required for update a DataProduct """ @@ -12307,6 +12328,16 @@ input UpdateOwnershipTypeInput { description: String } +""" +A standardized type of a user +""" +type DataHubPersona { + """ + The urn of the persona type + """ + urn: String! +} + """ Describes a generic filter on a dataset """ diff --git a/datahub-graphql-core/src/test/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolverTest.java b/datahub-graphql-core/src/test/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolverTest.java index 0cf4ed9896f89b..cf3c833cbba232 100644 --- a/datahub-graphql-core/src/test/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolverTest.java +++ b/datahub-graphql-core/src/test/java/com/linkedin/datahub/graphql/resolvers/assertion/ReportAssertionResultResolverTest.java @@ -61,8 +61,8 @@ public class ReportAssertionResultResolverTest { .setType(com.linkedin.assertion.AssertionResultErrorType.UNKNOWN_ERROR) .setProperties( new StringMap(Map.of("message", "an unknown error occurred")))) - .setExternalUrl(customAssertionUrl)) - .setRuntimeContext(new StringMap(Map.of("prop1", "value1"))); + .setExternalUrl(customAssertionUrl) + .setNativeResults(new StringMap(Map.of("prop1", "value1")))); @Test public void testGetSuccessReportAssertionRunEvent() throws Exception { @@ -91,8 +91,7 @@ public void testGetSuccessReportAssertionRunEvent() throws Exception { Mockito.eq(TEST_ASSERTION_URN), Mockito.eq(TEST_DATASET_URN), Mockito.eq(TEST_ASSERTION_RUN_EVENT.getTimestampMillis()), - Mockito.eq(TEST_ASSERTION_RUN_EVENT.getResult()), - Mockito.eq(TEST_ASSERTION_RUN_EVENT.getRuntimeContext())); + Mockito.eq(TEST_ASSERTION_RUN_EVENT.getResult())); } @Test @@ -126,7 +125,6 @@ public void testGetUpdateAssertionUnauthorized() throws Exception { Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any()); } @@ -146,7 +144,6 @@ public void testGetAssertionServiceException() { Mockito.any(), Mockito.any(), Mockito.any(), - Mockito.any(), Mockito.any()); ReportAssertionResultResolver resolver = new ReportAssertionResultResolver(mockService); diff --git a/datahub-web-react/package.json b/datahub-web-react/package.json index e2bc48164e77da..ca53932eba5189 100644 --- a/datahub-web-react/package.json +++ b/datahub-web-react/package.json @@ -71,7 +71,7 @@ "react-router-dom": "^5.3", "react-syntax-highlighter": "^15.4.4", "react-visibility-sensor": "^5.1.1", - "reactour": "1.18.7", + "reactour": "^1.19.3", "remirror": "^2.0.23", "styled-components": "^5.2.1", "turndown-plugin-gfm": "^1.0.2", diff --git a/datahub-web-react/src/app/entity/chart/ChartEntity.tsx b/datahub-web-react/src/app/entity/chart/ChartEntity.tsx index 2a54a4a96c6393..913d502972fe14 100644 --- a/datahub-web-react/src/app/entity/chart/ChartEntity.tsx +++ b/datahub-web-react/src/app/entity/chart/ChartEntity.tsx @@ -28,6 +28,7 @@ import { LOOKER_URN } from '../../ingest/source/builder/constants'; import { MatchedFieldList } from '../../search/matches/MatchedFieldList'; import { matchedInputFieldRenderer } from '../../search/matches/matchedInputFieldRenderer'; import { IncidentTab } from '../shared/tabs/Incident/IncidentTab'; +import { ChartQueryTab } from './ChartQueryTab'; /** * Definition of the DataHub Chart entity. @@ -110,6 +111,14 @@ export class ChartEntity implements Entity { component: ChartStatsSummarySubHeader, }} tabs={[ + { + name: 'Query', + component: ChartQueryTab, + display: { + visible: (_, chart: GetChartQuery) => (chart?.chart?.query?.rawQuery && true) || false, + enabled: (_, chart: GetChartQuery) => (chart?.chart?.query?.rawQuery && true) || false, + }, + }, { name: 'Documentation', component: DocumentationTab, diff --git a/datahub-web-react/src/app/entity/chart/ChartQueryTab.tsx b/datahub-web-react/src/app/entity/chart/ChartQueryTab.tsx new file mode 100644 index 00000000000000..7c28f4be88d8d5 --- /dev/null +++ b/datahub-web-react/src/app/entity/chart/ChartQueryTab.tsx @@ -0,0 +1,61 @@ +import { Typography } from 'antd'; +import React from 'react'; +import styled from 'styled-components'; +import { Prism as SyntaxHighlighter } from 'react-syntax-highlighter'; +import { GetChartQuery } from '../../../graphql/chart.generated'; +import { ANTD_GRAY } from '../shared/constants'; +import { useBaseEntity } from '../shared/EntityContext'; +import { InfoItem } from '../shared/components/styled/InfoItem'; + +const InfoSection = styled.div` + border-bottom: 1px solid ${ANTD_GRAY[4.5]}; + padding: 16px 20px; +`; + +const InfoItemContainer = styled.div<{ justifyContent }>` + display: flex; + position: relative; + justify-content: ${(props) => props.justifyContent}; + padding: 12px 2px; +`; + +const InfoItemContent = styled.div` + padding-top: 8px; +`; + +const QueryText = styled(Typography.Paragraph)` + margin-top: 20px; + background-color: ${ANTD_GRAY[2]}; +`; + +// NOTE: Yes, using `!important` is a shame. However, the SyntaxHighlighter is applying styles directly +// to the component, so there's no way around this +const NestedSyntax = styled(SyntaxHighlighter)` + background-color: transparent !important; + border: none !important; +`; + +export function ChartQueryTab() { + const baseEntity = useBaseEntity(); + const query = baseEntity?.chart?.query?.rawQuery || 'UNKNOWN'; + const type = baseEntity?.chart?.query?.type || 'UNKNOWN'; + + return ( + <> + + Details + + + {type.toUpperCase()} + + + + + Query + + {query} + + + + ); +} diff --git a/datahub-web-react/src/app/ingest/ManageIngestionPage.tsx b/datahub-web-react/src/app/ingest/ManageIngestionPage.tsx index baa92cbb7d46ce..1d04edbac228ac 100644 --- a/datahub-web-react/src/app/ingest/ManageIngestionPage.tsx +++ b/datahub-web-react/src/app/ingest/ManageIngestionPage.tsx @@ -58,9 +58,9 @@ export const ManageIngestionPage = () => { - Manage Ingestion + Manage Data Sources - Create, schedule, and run DataHub ingestion sources. + Configure and schedule syncs to import data from your data sources onClickTab(tab)}> diff --git a/datahub-web-react/src/app/ingest/source/IngestionSourceTable.tsx b/datahub-web-react/src/app/ingest/source/IngestionSourceTable.tsx index ad1e7f6425062a..00d04ed245edfa 100644 --- a/datahub-web-react/src/app/ingest/source/IngestionSourceTable.tsx +++ b/datahub-web-react/src/app/ingest/source/IngestionSourceTable.tsx @@ -3,7 +3,7 @@ import React from 'react'; import styled from 'styled-components/macro'; import { StyledTable } from '../../entity/shared/components/styled/StyledTable'; import { ANTD_GRAY } from '../../entity/shared/constants'; -import { CLI_EXECUTOR_ID } from './utils'; +import { CLI_EXECUTOR_ID, getIngestionSourceStatus } from './utils'; import { LastStatusColumn, TypeColumn, @@ -123,7 +123,7 @@ function IngestionSourceTable({ lastExecStatus: source.executions && source.executions?.executionRequests.length > 0 && - source.executions?.executionRequests[0].result?.status, + getIngestionSourceStatus(source.executions?.executionRequests[0].result), cliIngestion: source.config?.executorId === CLI_EXECUTOR_ID, })); diff --git a/datahub-web-react/src/app/ingest/source/builder/CreateScheduleStep.tsx b/datahub-web-react/src/app/ingest/source/builder/CreateScheduleStep.tsx index 3745ee0f44dc01..9207b9d0cfcf93 100644 --- a/datahub-web-react/src/app/ingest/source/builder/CreateScheduleStep.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/CreateScheduleStep.tsx @@ -10,6 +10,7 @@ import { TimezoneSelect } from './TimezoneSelect'; import { ANTD_GRAY, REDESIGN_COLORS } from '../../../entity/shared/constants'; import { lowerFirstLetter } from '../../../shared/textUtil'; import { IngestionSourceBuilderStep } from './steps'; +import { RequiredFieldForm } from '../../../shared/form/RequiredFieldForm'; const Section = styled.div` display: flex; @@ -31,10 +32,25 @@ const CronText = styled(Typography.Paragraph)` color: ${ANTD_GRAY[7]}; `; +const CronInput = styled(Input)` + margin-bottom: 8px; + max-width: 200px; +`; + +const Schedule = styled.div` + display: flex; + align-items: center; + justify-content: start; +`; + +const AdvancedSchedule = styled.div` + margin-left: 20px; +`; + const AdvancedCheckBox = styled(Typography.Text)` margin-right: 10px; - margin-bottom: 8px; `; + const CronSuccessCheck = styled(CheckCircleOutlined)` color: ${REDESIGN_COLORS.BLUE}; margin-right: 4px; @@ -123,9 +139,9 @@ export const CreateScheduleStep = ({ state, updateState, goTo, prev }: StepProps
Configure an Ingestion Schedule
-
+ Run on a schedule (Recommended) @@ -141,29 +157,31 @@ export const CreateScheduleStep = ({ state, updateState, goTo, prev }: StepProps )} Schedule}> -
- Advanced - setAdvancedCronCheck(event.target.checked)} - /> -
- {advancedCronCheck ? ( - setScheduleCronInterval(e.target.value)} - /> - ) : ( - - )} + + {advancedCronCheck ? ( + setScheduleCronInterval(e.target.value)} + /> + ) : ( + + )} + + Show Advanced + setAdvancedCronCheck(event.target.checked)} + /> + + {cronAsText.error && <>Invalid cron schedule. Cron must be of UNIX form:} {!cronAsText.text && ( @@ -183,7 +201,7 @@ export const CreateScheduleStep = ({ state, updateState, goTo, prev }: StepProps Choose a timezone for the schedule. - +
@@ -191,6 +209,7 @@ export const CreateScheduleStep = ({ state, updateState, goTo, prev }: StepProps data-testid="ingestion-schedule-next-button" disabled={!interval || interval.length === 0 || cronAsText.error} onClick={onClickNext} + type="primary" > Next diff --git a/datahub-web-react/src/app/ingest/source/builder/DataPlatformCard.tsx b/datahub-web-react/src/app/ingest/source/builder/DataPlatformCard.tsx new file mode 100644 index 00000000000000..34efbb30008294 --- /dev/null +++ b/datahub-web-react/src/app/ingest/source/builder/DataPlatformCard.tsx @@ -0,0 +1,68 @@ +import React from 'react'; +import { Button, Image } from 'antd'; +import styled from 'styled-components'; + +import { REDESIGN_COLORS } from '../../../entity/shared/constants'; + +const Container = styled(Button)` + padding: 32px; + height: 200px; + display: flex; + justify-content: center; + border-radius: 8px; + align-items: start; + flex-direction: column; + border: 1px solid #e0e0e0; + background-color: #ffffff; + &&:hover { + border: 1px solid ${REDESIGN_COLORS.BLUE}; + background-color: #ffffff; + } + white-space: unset; +`; + +const PlatformLogo = styled(Image)` + max-height: 32px; + height: 32px; + width: auto; + object-fit: contain; + background-color: transparent; +`; + +const LogoContainer = styled.div` + margin-bottom: 14px; +`; + +const Title = styled.div` + word-break: break-word; + color: #464646; + font-weight: bold; + font-size: 16px; + margin-bottom: 8px; +`; + +const Description = styled.div` + word-break: break-word; + text-align: left; + color: #7c7c7c; +`; + +type Props = { + logoUrl?: string; + logoComponent?: React.ReactNode; + name: string; + description?: string; + onClick?: () => void; +}; + +export const DataPlatformCard = ({ logoUrl, logoComponent, name, description, onClick }: Props) => { + return ( + + + {(logoUrl && ) || logoComponent} + + {name} + {description} + + ); +}; diff --git a/datahub-web-react/src/app/ingest/source/builder/DefineRecipeStep.tsx b/datahub-web-react/src/app/ingest/source/builder/DefineRecipeStep.tsx index 4ff4623b548c92..c16193b061b793 100644 --- a/datahub-web-react/src/app/ingest/source/builder/DefineRecipeStep.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/DefineRecipeStep.tsx @@ -164,7 +164,7 @@ export const DefineRecipeStep = ({ state, updateState, goTo, prev, ingestionSour - diff --git a/datahub-web-react/src/app/ingest/source/builder/IngestionDocumentationHint.tsx b/datahub-web-react/src/app/ingest/source/builder/IngestionDocumentationHint.tsx new file mode 100644 index 00000000000000..bda3d7f7424afd --- /dev/null +++ b/datahub-web-react/src/app/ingest/source/builder/IngestionDocumentationHint.tsx @@ -0,0 +1,68 @@ +import React from 'react'; +import styled from 'styled-components'; +import { Button, Tooltip } from 'antd'; +import { CloseOutlined } from '@ant-design/icons'; + +import { SourceConfig } from './types'; +import { ANTD_GRAY } from '../../../entity/shared/constants'; + +const Container = styled.div` + background-color: #ffffff; + border-radius: 8px; + padding: 12px 12px 16px 24px; + border: 1px solid #e0e0e0; + margin-bottom: 20px; +`; + +const Header = styled.div` + display: flex; + align-items: center; + justify-content: space-between; + margin-bottom: 12px; +`; + +const Title = styled.div` + font-size: 16px; + font-weight: bold; +`; + +const Description = styled.div` + font-size: 14px; + max-width: 90%; +`; + +const StyledCloseOutlined = styled(CloseOutlined)` + color: ${ANTD_GRAY[6]}; +`; + +interface Props { + sourceConfigs: SourceConfig; + onHide: () => void; +} + +export const IngestionDocumentationHint = ({ sourceConfigs, onHide }: Props) => { + const { displayName, docsUrl } = sourceConfigs; + return ( + +
+ Let's get connected! 🎉 + +
+ +
+ To import from {displayName}, we'll need some more information to connect to your instance. +
+
+ Check out the{' '} + + {displayName} Guide + {' '} + to understand the prerequisites, learn about available settings, and view examples to help connect + to the data source. +
+
+
+ ); +}; diff --git a/datahub-web-react/src/app/ingest/source/builder/IngestionSourceBuilderModal.tsx b/datahub-web-react/src/app/ingest/source/builder/IngestionSourceBuilderModal.tsx index 5a623b58af5c96..a41a8ec0f12ab9 100644 --- a/datahub-web-react/src/app/ingest/source/builder/IngestionSourceBuilderModal.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/IngestionSourceBuilderModal.tsx @@ -1,8 +1,7 @@ -import { Button, Modal, Steps, Typography } from 'antd'; +import { Modal, Steps, Typography } from 'antd'; import React, { useEffect, useRef, useState } from 'react'; import styled from 'styled-components'; import { isEqual } from 'lodash'; -import { ExpandAltOutlined, ShrinkOutlined } from '@ant-design/icons'; import { SourceBuilderState, StepProps } from './types'; import { CreateScheduleStep } from './CreateScheduleStep'; import { DefineRecipeStep } from './DefineRecipeStep'; @@ -10,15 +9,18 @@ import { NameSourceStep } from './NameSourceStep'; import { SelectTemplateStep } from './SelectTemplateStep'; import sourcesJson from './sources.json'; -const ExpandButton = styled(Button)` - && { - margin-right: 32px; +const StyledModal = styled(Modal)` + && .ant-modal-content { + border-radius: 16px; + overflow: hidden; + min-width: 400px; } `; const TitleContainer = styled.div` display: flex; justify-content: space-between; + border-radius: 12px; `; const StepsContainer = styled.div` @@ -31,9 +33,9 @@ const StepsContainer = styled.div` * Mapping from the step type to the title for the step */ export enum IngestionSourceBuilderStepTitles { - SELECT_TEMPLATE = 'Choose Type', - DEFINE_RECIPE = 'Configure Recipe', - CREATE_SCHEDULE = 'Schedule Ingestion', + SELECT_TEMPLATE = 'Choose Data Source', + DEFINE_RECIPE = 'Configure Connection', + CREATE_SCHEDULE = 'Sync Schedule', NAME_SOURCE = 'Finish up', } @@ -57,6 +59,8 @@ export enum IngestionSourceBuilderStep { NAME_SOURCE = 'NAME_SOURCE', } +const modalBodyStyle = { padding: '16px 24px 16px 24px', backgroundColor: '#F6F6F6' }; + type Props = { initialState?: SourceBuilderState; visible: boolean; @@ -66,14 +70,17 @@ type Props = { export const IngestionSourceBuilderModal = ({ initialState, visible, onSubmit, onCancel }: Props) => { const isEditing = initialState !== undefined; - const titleText = isEditing ? 'Edit Ingestion Source' : 'New Ingestion Source'; + const titleText = isEditing ? 'Edit Data Source' : 'Connect Data Source'; const initialStep = isEditing ? IngestionSourceBuilderStep.DEFINE_RECIPE : IngestionSourceBuilderStep.SELECT_TEMPLATE; const [stepStack, setStepStack] = useState([initialStep]); - const [modalExpanded, setModalExpanded] = useState(false); - const [ingestionBuilderState, setIngestionBuilderState] = useState({}); + const [ingestionBuilderState, setIngestionBuilderState] = useState({ + schedule: { + interval: '0 0 * * *', + }, + }); const ingestionSources = JSON.parse(JSON.stringify(sourcesJson)); // TODO: replace with call to server once we have access to dynamic list of sources @@ -122,28 +129,28 @@ export const IngestionSourceBuilderModal = ({ initialState, visible, onSubmit, o const StepComponent: React.FC = IngestionSourceBuilderStepComponent[currentStep]; return ( - {titleText} - setModalExpanded(!modalExpanded)}> - {(modalExpanded && ) || } - } style={{ top: 40 }} + bodyStyle={modalBodyStyle} visible={visible} onCancel={onCancel} > - - - {Object.keys(IngestionSourceBuilderStep).map((item) => ( - - ))} - - + {currentStepIndex > 0 ? ( + + + {Object.keys(IngestionSourceBuilderStep).map((item) => ( + + ))} + + + ) : null} - + ); }; diff --git a/datahub-web-react/src/app/ingest/source/builder/NameSourceStep.tsx b/datahub-web-react/src/app/ingest/source/builder/NameSourceStep.tsx index 5573e5a3e39040..898fbd6a6d9268 100644 --- a/datahub-web-react/src/app/ingest/source/builder/NameSourceStep.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/NameSourceStep.tsx @@ -1,7 +1,8 @@ -import { Button, Checkbox, Collapse, Form, Input, Typography } from 'antd'; +import { Button, Checkbox, Collapse, Form, Input, Tooltip, Typography } from 'antd'; import React from 'react'; import styled from 'styled-components'; import { SourceBuilderState, StepProps, StringMapEntryInput } from './types'; +import { RequiredFieldForm } from '../../../shared/form/RequiredFieldForm'; const ControlsContainer = styled.div` display: flex; @@ -156,7 +157,7 @@ export const NameSourceStep = ({ state, updateState, prev, submit }: StepProps) return ( <> -
+ - Give this ingestion source a name. + Give this data source a name - +
@@ -263,13 +264,15 @@ export const NameSourceStep = ({ state, updateState, prev, submit }: StepProps) > Save - + + +
diff --git a/datahub-web-react/src/app/ingest/source/builder/RecipeBuilder.tsx b/datahub-web-react/src/app/ingest/source/builder/RecipeBuilder.tsx index 880420386fa67e..2d0bfe340c5065 100644 --- a/datahub-web-react/src/app/ingest/source/builder/RecipeBuilder.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/RecipeBuilder.tsx @@ -10,6 +10,7 @@ import { SourceBuilderState, SourceConfig } from './types'; import { CSV, LOOKER, LOOK_ML } from './constants'; import { LookerWarning } from './LookerWarning'; import { CSVInfo } from './CSVInfo'; +import { IngestionDocumentationHint } from './IngestionDocumentationHint'; export const ControlsContainer = styled.div` display: flex; @@ -66,6 +67,7 @@ function RecipeBuilder(props: Props) { const { state, isEditing, displayRecipe, sourceConfigs, setStagedRecipe, onClickNext, goToPrevious } = props; const { type } = state; const [isViewingForm, setIsViewingForm] = useState(true); + const [hideDocsHint, setHideDocsHint] = useState(false); function switchViews(isFormView: boolean) { try { @@ -81,12 +83,14 @@ function RecipeBuilder(props: Props) { return (
+ {!hideDocsHint && isViewingForm && sourceConfigs ? ( + setHideDocsHint(true)} sourceConfigs={sourceConfigs} /> + ) : null} {(type === LOOKER || type === LOOK_ML) && } {type === CSV && } - - {sourceConfigs?.displayName} Recipe + {sourceConfigs?.displayName} Details Previous - diff --git a/datahub-web-react/src/app/ingest/source/builder/RecipeForm/RecipeForm.tsx b/datahub-web-react/src/app/ingest/source/builder/RecipeForm/RecipeForm.tsx index bdee01d6498ee7..4199658568b9a6 100644 --- a/datahub-web-react/src/app/ingest/source/builder/RecipeForm/RecipeForm.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/RecipeForm/RecipeForm.tsx @@ -1,9 +1,11 @@ -import { Button, Collapse, Form, message, Tooltip, Typography } from 'antd'; import React, { Fragment } from 'react'; + +import { Button, Collapse, Form, message, Tooltip, Typography } from 'antd'; import { get } from 'lodash'; import YAML from 'yamljs'; import { ApiOutlined, FilterOutlined, QuestionCircleOutlined, SettingOutlined } from '@ant-design/icons'; import styled from 'styled-components/macro'; + import { jsonToYaml } from '../../utils'; import { CONNECTORS_WITH_TEST_CONNECTION, RecipeSections, RECIPE_FIELDS } from './constants'; import FormField from './FormField'; @@ -11,6 +13,7 @@ import TestConnectionButton from './TestConnection/TestConnectionButton'; import { useListSecretsQuery } from '../../../../../graphql/ingestion.generated'; import { RecipeField, setFieldValueOnRecipe } from './common'; import { SourceBuilderState, SourceConfig } from '../types'; +import { RequiredFieldForm } from '../../../../shared/form/RequiredFieldForm'; export const ControlsContainer = styled.div` display: flex; @@ -140,7 +143,7 @@ function RecipeForm(props: Props) { } return ( -
} - text="Advanced" + text="Settings" sectionTooltip={advancedSectionTooltip} /> } @@ -230,9 +233,11 @@ function RecipeForm(props: Props) { - + -
+ ); } diff --git a/datahub-web-react/src/app/ingest/source/builder/RecipeForm/common.tsx b/datahub-web-react/src/app/ingest/source/builder/RecipeForm/common.tsx index 43d899301c2fc7..cbaf2f4d87991f 100644 --- a/datahub-web-react/src/app/ingest/source/builder/RecipeForm/common.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/RecipeForm/common.tsx @@ -276,7 +276,7 @@ export const INCLUDE_LINEAGE: RecipeField = { export const INCLUDE_TABLE_LINEAGE: RecipeField = { name: 'include_table_lineage', label: 'Include Table Lineage', - tooltip: 'Extract Tabel-Level lineage metadata. Enabling this may increase the duration of the extraction process.', + tooltip: 'Extract Tabel-Level lineage metadata. Enabling this may increase the duration of the sync.', type: FieldType.BOOLEAN, fieldPath: 'source.config.include_table_lineage', rules: null, @@ -286,8 +286,7 @@ const isProfilingEnabledFieldPath = 'source.config.profiling.enabled'; export const TABLE_PROFILING_ENABLED: RecipeField = { name: 'profiling.enabled', label: 'Enable Table Profiling', - tooltip: - 'Generate Data Profiles for extracted Tables. Enabling this may increase the duration of the extraction process.', + tooltip: 'Generate Data Profiles for extracted Tables. Enabling this may increase the duration of the sync.', type: FieldType.BOOLEAN, fieldPath: isProfilingEnabledFieldPath, rules: null, @@ -298,7 +297,7 @@ export const COLUMN_PROFILING_ENABLED: RecipeField = { name: 'column_profiling.enabled', label: 'Enable Column Profiling', tooltip: - 'Generate Data Profiles for the Columns in extracted Tables. Enabling this may increase the duration of the extraction process.', + 'Generate Data Profiles for the Columns in extracted Tables. Enabling this may increase the duration of the sync.', type: FieldType.BOOLEAN, fieldPath: isTableProfilingOnlyFieldPath, rules: null, @@ -466,7 +465,7 @@ export const START_TIME: RecipeField = { name: 'start_time', label: 'Start Time', tooltip: - 'Earliest date used when processing audit logs for lineage, usage, and more. Default: Last full day in UTC or last time DataHub ingested usage (if stateful ingestion is enabled). Tip: Set this to an older date (e.g. 1 month ago) to bootstrap your first ingestion run, and then reduce for subsequent runs. Changing this may increase the duration of the extraction process.', + 'Earliest date used when processing audit logs for lineage, usage, and more. Default: Last full day in UTC or last time DataHub ingested usage (if stateful ingestion is enabled). Tip: Set this to an older date (e.g. 1 month ago) to bootstrap your first ingestion run, and then reduce for subsequent runs. Changing this may increase the duration of the sync.', placeholder: 'Select date and time', type: FieldType.DATE, fieldPath: startTimeFieldPath, diff --git a/datahub-web-react/src/app/ingest/source/builder/SelectTemplateStep.tsx b/datahub-web-react/src/app/ingest/source/builder/SelectTemplateStep.tsx index 6b771d459c4ef9..3998915e07a2ce 100644 --- a/datahub-web-react/src/app/ingest/source/builder/SelectTemplateStep.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/SelectTemplateStep.tsx @@ -1,39 +1,60 @@ +import React, { useState } from 'react'; + import { Button, Input } from 'antd'; import { FormOutlined, SearchOutlined } from '@ant-design/icons'; -import React, { useState } from 'react'; import styled from 'styled-components'; -import { LogoCountCard } from '../../../shared/LogoCountCard'; import { SourceConfig, SourceBuilderState, StepProps } from './types'; import { IngestionSourceBuilderStep } from './steps'; import useGetSourceLogoUrl from './useGetSourceLogoUrl'; import { CUSTOM } from './constants'; import { ANTD_GRAY } from '../../../entity/shared/constants'; +import { DataPlatformCard } from './DataPlatformCard'; -const Section = styled.div` +const Container = styled.div` + max-height: 82vh; display: flex; flex-direction: column; - padding-bottom: 12px; `; -const PlatformListContainer = styled.div` +const Section = styled.div` display: flex; - justify-content: left; - align-items: center; - flex-wrap: wrap; + flex-direction: column; + padding-bottom: 12px; + overflow: hidden; `; const CancelButton = styled(Button)` - && { - margin-left: 12px; - } + max-width: 120px; +`; + +const SearchBarContainer = styled.div` + display: flex; + justify-content: end; + width: auto; + padding-right: 12px; `; const StyledSearchBar = styled(Input)` background-color: white; - border-radius: 70px; + border-radius: 8px; box-shadow: 0px 0px 30px 0px rgb(239 239 239); - width: 45%; - margin: 0 0 15px 12px; + border: 1px solid #e0e0e0; + margin: 0 0 15px 0px; + max-width: 300px; + font-size: 16px; +`; + +const StyledSearchOutlined = styled(SearchOutlined)` + color: #a9adbd; +`; + +const PlatformListContainer = styled.div` + display: grid; + grid-template-columns: repeat(auto-fill, minmax(min(100%, 31%), 1fr)); + gap: 10px; + height: 100%; + overflow-y: auto; + padding-right: 12px; `; interface SourceOptionProps { @@ -42,7 +63,7 @@ interface SourceOptionProps { } function SourceOption({ source, onClick }: SourceOptionProps) { - const { name, displayName } = source; + const { name, displayName, description } = source; const logoUrl = useGetSourceLogoUrl(name); let logoComponent; @@ -50,7 +71,15 @@ function SourceOption({ source, onClick }: SourceOptionProps) { logoComponent = ; } - return ; + return ( + + ); } /** @@ -76,22 +105,24 @@ export const SelectTemplateStep = ({ state, updateState, goTo, cancel, ingestion ); return ( - <> +
- setSearchFilter(e.target.value)} - allowClear - prefix={} - /> - + + setSearchFilter(e.target.value)} + allowClear + prefix={} + /> + + {filteredSources.map((source) => ( onSelectTemplate(source.name)} /> ))}
Cancel - +
); }; diff --git a/datahub-web-react/src/app/ingest/source/builder/TimezoneSelect.tsx b/datahub-web-react/src/app/ingest/source/builder/TimezoneSelect.tsx index d9f3df1fc99299..21731b69cf46b6 100644 --- a/datahub-web-react/src/app/ingest/source/builder/TimezoneSelect.tsx +++ b/datahub-web-react/src/app/ingest/source/builder/TimezoneSelect.tsx @@ -1,21 +1,28 @@ import { Select } from 'antd'; import React from 'react'; import moment from 'moment-timezone'; +import styled from 'styled-components'; + +const StyledSelect = styled(Select)` + max-width: 300px; +`; type Props = { value: string; - onChange: (newTimezone: string) => void; + onChange: (newTimezone: any) => void; }; export const TimezoneSelect = ({ value, onChange }: Props) => { const timezones = moment.tz.names(); return ( <> - + ); }; diff --git a/datahub-web-react/src/app/ingest/source/builder/sources.json b/datahub-web-react/src/app/ingest/source/builder/sources.json index d4faf82a20605a..c35a7a033a8ab3 100644 --- a/datahub-web-react/src/app/ingest/source/builder/sources.json +++ b/datahub-web-react/src/app/ingest/source/builder/sources.json @@ -3,55 +3,63 @@ "urn": "urn:li:dataPlatform:bigquery", "name": "bigquery", "displayName": "BigQuery", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/bigquery/", + "description": "Import Projects, Datasets, Tables, Views, lineage, queries, and statistics from BigQuery.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/bigquery/overview", "recipe": "source:\n type: bigquery\n config:\n include_table_lineage: true\n include_usage_statistics: true\n include_tables: true\n include_views: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:redshift", "name": "redshift", "displayName": "Redshift", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/redshift/", + "description": "Import Tables, Views, Databases, Schemas, lineage, queries, and statistics from Redshift.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/redshift/overview", "recipe": "source: \n type: redshift\n config:\n # Coordinates\n host_port: # Your Redshift host and post, e.g. example.something.us-west-2.redshift.amazonaws.com:5439\n database: # Your Redshift database, e.g. SampleDatabase\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: null # Your Redshift username, e.g. admin\n\n table_lineage_mode: stl_scan_based\n include_table_lineage: true\n include_tables: true\n include_views: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:snowflake", "name": "snowflake", "displayName": "Snowflake", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/snowflake/", + "description": "Import Tables, Views, Databases, Schemas, lineage, queries, and statistics from Snowflake.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/snowflake/overview", "recipe": "source: \n type: snowflake\n config:\n account_id: null\n include_table_lineage: true\n include_view_lineage: true\n include_tables: true\n include_views: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, { - "urn": "urn:li:dataPlatform:kafka", - "name": "kafka", - "displayName": "Kafka", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/kafka/", - "recipe": "source:\n type: kafka\n config:\n connection:\n consumer_config:\n security.protocol: \"PLAINTEXT\"\n stateful_ingestion:\n enabled: false" + "urn": "urn:li:dataPlatform:unity-catalog", + "name": "unity-catalog", + "displayName": "Databricks", + "description": "Import Metastores, Schemas, Tables, lineage, queries, and statistics from Databricks Unity Catalog.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/databricks/#module-unity-catalog", + "recipe": "source:\n type: unity-catalog\n config:\n # Coordinates\n workspace_url: null\n include_table_lineage: true\n include_column_lineage: false\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:looker", "name": "looker", "displayName": "Looker", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/looker/", - "recipe": "source:\n type: looker\n config:\n # Coordinates\n base_url: # Your Looker instance URL, e.g. https://company.looker.com:19999\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n client_id: null # Your Looker client id, e.g. admin\n stateful_ingestion:\n enabled: true" + "description": "Import Models, Explores, Views, Looks, Dashboards, and lineage from Looker.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/looker/overview#looker", + "recipe": "source:\n type: looker\n config:\n # Coosrdinates\n base_url: # Your Looker instance URL, e.g. https://company.looker.com:19999\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n client_id: null # Your Looker client id, e.g. admin\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:lookml", "name": "lookml", "displayName": "LookML", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/looker/#module-lookml", + "description": "Import Models, Explores, Views, Looks, Dashboards, and lineage from LookML files.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/looker/overview#lookml", "recipe": "source:\n type: lookml\n config:\n parse_table_names_from_sql: true\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:tableau", "name": "tableau", "displayName": "Tableau", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/tableau/", + "description": "Import Data Sources, Workbooks, Worksheets, Tags, Dashboards, and lineage from Tableau.", + "docsUrl": "https://datahubproject.io/docs/quick-ingestion-guides/tableau/overview", "recipe": "source:\n type: tableau\n config:\n # Coordinates\n connect_uri: null\n stateful_ingestion:\n enabled: true" }, { "urn": "urn:li:dataPlatform:powerbi", "name": "powerbi", "displayName": "PowerBI", + "description": "Import Dashboards, Tiles, Datasets, and lineage from PowerBI.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/powerbi/", "recipe": "source:\n type: \"powerbi\"\n config:\n # Your Power BI tenant identifier\n tenant_id: null\n # Your Power BI client id\n client_id: null\n # Your Power BI client secret\n client_secret: null\n stateful_ingestion:\n enabled: true" }, @@ -59,6 +67,7 @@ "urn": "urn:li:dataPlatform:dbt", "name": "dbt-cloud", "displayName": "dbt Cloud", + "description": "Import Sources, Seeds, Models, Snapshots, Tests, and lineage from dbt cloud.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/dbt/#module-dbt-cloud", "recipe": "source:\n type: dbt-cloud\n config:\n account_id: null\n project_id: null\n job_id: null\n target_platform: null\n stateful_ingestion:\n enabled: true" }, @@ -66,6 +75,7 @@ "urn": "urn:li:dataPlatform:mysql", "name": "mysql", "displayName": "MySQL", + "description": "Import Tables, Views, Databases, Schemas, and statistics from MySQL.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mysql/", "recipe": "source: \n type: mysql\n config: \n # Coordinates\n host_port: # Your MySQL host and post, e.g. mysql:3306\n database: # Your MySQL database name, e.g. datahub\n \n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: null # Your MySQL username, e.g. admin\n\n # Options\n include_tables: true\n include_views: true\n\n # Profiling\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, @@ -73,13 +83,23 @@ "urn": "urn:li:dataPlatform:postgres", "name": "postgres", "displayName": "Postgres", + "description": "Import Tables, Views, Databases, Schemas, and statistics from Postgres.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/postgres/", "recipe": "source: \n type: postgres\n config:\n # Coordinates\n host_port: # Your Postgres host and port, e.g. postgres:5432\n database: # Your Postgres Database, e.g. sample_db\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: null # Your Postgres username, e.g. admin\n\n # Options\n include_tables: true\n include_views: true\n\n # Profiling\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, + { + "urn": "urn:li:dataPlatform:kafka", + "name": "kafka", + "displayName": "Kafka", + "description": "Import streaming topics from Kafka.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/kafka/", + "recipe": "source:\n type: kafka\n config:\n connection:\n consumer_config:\n security.protocol: \"PLAINTEXT\"\n stateful_ingestion:\n enabled: false" + }, { "urn": "urn:li:dataPlatform:hive", "name": "hive", "displayName": "Hive", + "description": "Import Tables, Views, Databases, Schemas, and statistics from Hive.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/hive/", "recipe": "source: \n type: hive\n config:\n # Coordinates\n host_port: # Your Hive host and port, e.g. hive:10000\n database: # Your Hive database name, e.g. SampleDatabase (Optional, if not specified, ingests from all databases)\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: null # Your Hive username, e.g. admin\n stateful_ingestion:\n enabled: true" }, @@ -87,6 +107,7 @@ "urn": "urn:li:dataPlatform:presto", "name": "presto", "displayName": "Presto", + "description": "Import Tables, Databases, Schemas, and statistics from Presto.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/presto/", "recipe": "source:\n type: presto\n config:\n # Coordinates\n host_port: null\n # The name of the catalog from getting the usage\n database: null\n # Credentials\n username: null\n include_views: true\n include_tables: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, @@ -94,13 +115,23 @@ "urn": "urn:li:dataPlatform:trino", "name": "trino", "displayName": "Trino", + "description": "Import Tables, Databases, Schemas, and statistics from Trino.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/trino/", "recipe": "source:\n type: trino\n config:\n # Coordinates\n host_port: null\n # The name of the catalog from getting the usage\n database: null\n # Credentials\n username: null\n include_views: true\n include_tables: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, + { + "urn": "urn:li:dataPlatform:glue", + "name": "glue", + "displayName": "Glue", + "description": "Import Tables, Databases, Jobs, statistics, and lineage to S3 from AWS Glue.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/glue/", + "recipe": "source:\n type: glue\n config:\n # AWS credentials. \n aws_region: # The region for your AWS Glue instance. \n # Add secret in Secrets Tab with relevant names for each variable\n # The access key for your AWS account.\n aws_access_key_id: \"${AWS_ACCESS_KEY_ID}\"\n # The secret key for your AWS account.\n aws_secret_access_key: \"${AWS_SECRET_KEY}\"\n aws_session_token: # The session key for your AWS account. This is only needed when you are using temporary credentials.\n # aws_role: # (Optional) The role to assume (Role chaining supported by using a sorted list).\n\n # Allow / Deny specific databases & tables\n # database_pattern:\n # allow:\n # - \"flights-database\"\n # table_pattern:\n # allow:\n # - \"avro\"" + }, { "urn": "urn:li:dataPlatform:mssql", "name": "mssql", "displayName": "Microsoft SQL Server", + "description": "Import Tables, Views, Databases, Schemas, and statistics from SQL Server.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mssql/", "recipe": "source:\n type: mssql\n config:\n # Coordinates\n host_port: null\n # The name\n database: null\n # Credentials\n username: null\n include_views: true\n include_tables: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, @@ -108,20 +139,15 @@ "urn": "urn:li:dataPlatform:mariadb", "name": "mariadb", "displayName": "MariaDB", + "description": "Import Tables, Views, Databases, Schemas, and statistics from MariaDB.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mariadb/", "recipe": "source:\n type: mariadb\n config:\n # Coordinates\n host_port: null\n # The name\n database: null\n # Credentials\n username: null\n include_views: true\n include_tables: true\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" }, - { - "urn": "urn:li:dataPlatform:unity-catalog", - "name": "unity-catalog", - "displayName": "Databricks Unity Catalog", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/databricks/#module-unity-catalog", - "recipe": "source:\n type: unity-catalog\n config:\n # Coordinates\n workspace_url: null\n include_table_lineage: true\n include_column_lineage: false\n stateful_ingestion:\n enabled: true" - }, { "urn": "urn:li:dataPlatform:mongodb", "name": "mongodb", "displayName": "MongoDB", + "description": "Import Databases and Collections from MongoDB.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mongodb/", "recipe": "source:\n type: mongodb\n config:\n # Coordinates\n connect_uri: # Your MongoDB connect URI, e.g. \"mongodb://localhost\"\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: \"${MONGO_USERNAME}\" # Your MongoDB username, e.g. admin\n password: \"${MONGO_PASSWORD}\" # Your MongoDB password, e.g. password_01\n\n # Options (recommended)\n enableSchemaInference: True\n useRandomSampling: True\n maxSchemaSize: 300" }, @@ -129,20 +155,15 @@ "urn": "urn:li:dataPlatform:dynamodb", "name": "dynamodb", "displayName": "DynamoDB", + "description": "Import Tables from DynamoDB.", "docsUrl": "https://datahubproject.io/docs/metadata-ingestion/", "recipe": "source:\n type: dynamodb\n config:\n platform_instance: \"AWS_ACCOUNT_ID\"\n aws_access_key_id : '${AWS_ACCESS_KEY_ID}'\n aws_secret_access_key : '${AWS_SECRET_ACCESS_KEY}'\n # If there are items that have most representative fields of the table, users could use the\n # `include_table_item` option to provide a list of primary keys of the table in dynamodb format.\n # For each `region.table`, the list of primary keys can be at most 100.\n # We include these items in addition to the first 100 items in the table when we scan it.\n # include_table_item:\n # region.table_name:\n # [\n # {\n # 'partition_key_name': { 'attribute_type': 'attribute_value' },\n # 'sort_key_name': { 'attribute_type': 'attribute_value' },\n # },\n # ]" }, - { - "urn": "urn:li:dataPlatform:glue", - "name": "glue", - "displayName": "Glue", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/glue/", - "recipe": "source:\n type: glue\n config:\n # AWS credentials. \n aws_region: # The region for your AWS Glue instance. \n # Add secret in Secrets Tab with relevant names for each variable\n # The access key for your AWS account.\n aws_access_key_id: \"${AWS_ACCESS_KEY_ID}\"\n # The secret key for your AWS account.\n aws_secret_access_key: \"${AWS_SECRET_KEY}\"\n aws_session_token: # The session key for your AWS account. This is only needed when you are using temporary credentials.\n # aws_role: # (Optional) The role to assume (Role chaining supported by using a sorted list).\n\n # Allow / Deny specific databases & tables\n # database_pattern:\n # allow:\n # - \"flights-database\"\n # table_pattern:\n # allow:\n # - \"avro\"" - }, { "urn": "urn:li:dataPlatform:oracle", "name": "oracle", "displayName": "Oracle", + "description": "Import Databases, Schemas, Tables, Views, statistics, and lineage from Oracle.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/oracle/", "recipe": "source: \n type: oracle\n config:\n # Coordinates\n host_port: # Your Oracle host and port, e.g. oracle:5432\n database: # Your Oracle database name, e.g. sample_db\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: \"${ORACLE_USERNAME}\" # Your Oracle username, e.g. admin\n password: \"${ORACLE_PASSWORD}\" # Your Oracle password, e.g. password_01\n\n # Optional service name\n # service_name: # Your service name, e.g. svc # omit database if using this option" }, @@ -150,6 +171,7 @@ "urn": "urn:li:dataPlatform:superset", "name": "superset", "displayName": "Superset", + "description": "Import Charts and Dashboards from Superset", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/superset/", "recipe": "source:\n type: superset\n config:\n # Coordinates\n connect_uri: http://localhost:8088\n\n # Credentials\n username: user\n password: pass\n provider: ldap" }, @@ -157,6 +179,7 @@ "urn": "urn:li:dataPlatform:athena", "name": "athena", "displayName": "Athena", + "description": "Import Schemas, Tables, Views, and lineage to S3 from Athena.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/athena/", "recipe": "source:\n type: athena\n config:\n # Coordinates\n aws_region: my_aws_region\n work_group: primary\n\n # Options\n s3_staging_dir: \"s3://my_staging_athena_results_bucket/results/\"" }, @@ -164,6 +187,7 @@ "urn": "urn:li:dataPlatform:clickhouse", "name": "clickhouse", "displayName": "ClickHouse", + "description": "Import Tables, Views, Materialized Views, Dictionaries, statistics, queries, and lineage from ClickHouse.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/clickhouse/", "recipe": "source:\n type: clickhouse\n config:\n # Coordinates\n host_port: localhost:9000\n\n # Credentials\n username: user\n password: pass\n\n # Options\n platform_instance: DatabaseNameToBeIngested\n\n include_views: true # whether to include views, defaults to True\n include_tables: true # whether to include views, defaults to True\n\nsink:\n # sink configs\n\n#---------------------------------------------------------------------------\n# For the HTTP interface:\n#---------------------------------------------------------------------------\nsource:\n type: clickhouse\n config:\n host_port: localhost:8443\n protocol: https\n\n#---------------------------------------------------------------------------\n# For the Native interface:\n#---------------------------------------------------------------------------\n\nsource:\n type: clickhouse\n config:\n host_port: localhost:9440\n scheme: clickhouse+native\n secure: True" }, @@ -171,13 +195,23 @@ "urn": "urn:li:dataPlatform:druid", "name": "druid", "displayName": "Druid", + "description": "Import Databases, Schemas, Tables, statistics, and lineage from Druid.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/druid/", "recipe": "source:\n type: druid\n config:\n # Coordinates\n host_port: \"localhost:8082\"\n\n # Credentials\n username: admin\n password: password" }, + { + "urn": "urn:li:dataPlatform:mode", + "name": "mode", + "displayName": "Mode", + "description": "Import Reports, Charts, and lineage from Mode.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mode/", + "recipe": "source:\n type: mode\n config:\n # Coordinates\n connect_uri: http://app.mode.com\n\n # Credentials\n token: token\n password: pass\n\n # Options\n workspace: \"datahub\"\n default_schema: \"public\"\n owner_username_instead_of_email: False\n api_options:\n retry_backoff_multiplier: 2\n max_retry_interval: 10\n max_attempts: 5" + }, { "urn": "urn:li:dataPlatform:metabase", "name": "metabase", "displayName": "Metabase", + "description": "Import Collections, Dashboards, and Charts from Metabase.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/metabase/", "recipe": "source:\n type: metabase\n config:\n # Coordinates\n connect_uri:\n\n # Credentials\n username: root\n password: example" }, @@ -185,20 +219,15 @@ "urn": "urn:li:dataPlatform:mlflow", "name": "mlflow", "displayName": "MLflow", + "description": "Import Registered Models, Model Versions, and Model Stages from MLflow.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mlflow/", "recipe": "source:\n type: mlflow\n config:\n tracking_uri: tracking_uri" }, - { - "urn": "urn:li:dataPlatform:mode", - "name": "mode", - "displayName": "Mode", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/mode/", - "recipe": "source:\n type: mode\n config:\n # Coordinates\n connect_uri: http://app.mode.com\n\n # Credentials\n token: token\n password: pass\n\n # Options\n workspace: \"datahub\"\n default_schema: \"public\"\n owner_username_instead_of_email: False\n api_options:\n retry_backoff_multiplier: 2\n max_retry_interval: 10\n max_attempts: 5" - }, { "urn": "urn:li:dataPlatform:azure-ad", "name": "azure-ad", "displayName": "Azure AD", + "description": "Import Users and Groups from Azure Active Directory.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/azure-ad/", "recipe": "source:\n type: azure-ad\n config:\n client_id: # Your Azure Client ID, e.g. \"00000000-0000-0000-0000-000000000000\"\n tenant_id: # Your Azure Tenant ID, e.g. \"00000000-0000-0000-0000-000000000000\"\n # Add secret in Secrets Tab with this name\n client_secret: \n redirect: # Your Redirect URL, e.g. \"https://login.microsoftonline.com/common/oauth2/nativeclient\"\n authority: # Your Authority URL, e.g. \"https://login.microsoftonline.com/00000000-0000-0000-0000-000000000000\"\n token_url: # Your Token URL, e.g. \"https://login.microsoftonline.com/00000000-0000-0000-0000-000000000000/oauth2/token\"\n graph_url: # The Graph URL, e.g. \"https://graph.microsoft.com/v1.0\"\n \n # Optional flags to ingest users, groups, or both\n ingest_users: True\n ingest_groups: True\n \n # Optional Allow / Deny extraction of particular Groups\n # groups_pattern:\n # allow:\n # - \".*\"\n\n # Optional Allow / Deny extraction of particular Users.\n # users_pattern:\n # allow:\n # - \".*\"" }, @@ -206,6 +235,7 @@ "urn": "urn:li:dataPlatform:okta", "name": "okta", "displayName": "Okta", + "description": "Import Users and Groups from Okta.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/okta/", "recipe": "source:\n type: okta\n config:\n # Coordinates\n okta_domain: # Your Okta Domain, e.g. \"dev-35531955.okta.com\"\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n okta_api_token: # Your Okta API Token, e.g. \"11be4R_M2MzDqXawbTHfKGpKee0kuEOfX1RCQSRx99\"\n\n # Optional flags to ingest users, groups, or both\n ingest_users: True\n ingest_groups: True\n\n # Optional: Customize the mapping to DataHub Username from an attribute appearing in the Okta User\n # profile. Reference: https://developer.okta.com/docs/reference/api/users/\n # okta_profile_to_username_attr: str = \"login\"\n # okta_profile_to_username_regex: str = \"([^@]+)\"\n \n # Optional: Customize the mapping to DataHub Group from an attribute appearing in the Okta Group\n # profile. Reference: https://developer.okta.com/docs/reference/api/groups/\n # okta_profile_to_group_name_attr: str = \"name\"\n # okta_profile_to_group_name_regex: str = \"(.*)\"\n \n # Optional: Include deprovisioned or suspended Okta users in the ingestion.\n # include_deprovisioned_users = False\n # include_suspended_users = False" }, @@ -213,6 +243,7 @@ "urn": "urn:li:dataPlatform:vertica", "name": "vertica", "displayName": "Vertica", + "description": "Import Databases, Schemas, Tables, Views, Projections, statistics, and lineage from Vertica.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/vertica/", "recipe": "source:\n type: vertica\n config:\n # Coordinates\n host_port: localhost:5433\n # The name of the vertica database\n database: Database_Name\n # Credentials\n username: Vertica_User\n password: Vertica_Password\n\n include_tables: true\n include_views: true\n include_projections: true\n include_models: true\n include_view_lineage: true\n include_projection_lineage: true\n profiling:\n enabled: false\n stateful_ingestion:\n enabled: true " }, @@ -220,42 +251,48 @@ "urn": "urn:li:dataPlatform:fivetran", "name": "fivetran", "displayName": "Fivetran", + "description": "Import Connectors, Destinations, Sync Histor, Users, and lineage from FiveTran.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/fivetran/", "recipe": "source:\n type: fivetran\n config:\n # Fivetran log connector destination server configurations\n fivetran_log_config:\n destination_platform: snowflake\n snowflake_destination_config:\n # Coordinates\n account_id: snowflake_account_id\n warehouse: warehouse_name\n database: snowflake_db\n log_schema: fivetran_log_schema\n\n # Credentials\n username: ${SNOWFLAKE_USER}\n password: ${SNOWFLAKE_PASS}\n role: snowflake_role\n\n # Optional - filter for certain connector names instead of ingesting everything.\n # connector_patterns:\n # allow:\n # - connector_name\n\n # Optional -- This mapping is optional and only required to configure platform-instance for source\n # A mapping of Fivetran connector id to data platform instance\n # sources_to_platform_instance:\n # calendar_elected:\n # platform_instance: cloud_postgres_instance\n # env: DEV\n\n # Optional -- This mapping is optional and only required to configure platform-instance for destination.\n # A mapping of Fivetran destination id to data platform instance\n # destination_to_platform_instance:\n # calendar_elected:\n # platform_instance: cloud_postgres_instance\n # env: DEV" }, { - "urn": "urn:li:dataPlatform:csv-enricher", - "name": "csv-enricher", - "displayName": "CSV", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/csv'", - "recipe": "source: \n type: csv-enricher \n config: \n # URL of your csv file to ingest \n filename: \n array_delimiter: '|' \n delimiter: ',' \n write_semantics: PATCH" - }, - { - "urn": "urn:li:dataPlatform:custom", - "name": "custom", - "displayName": "Other", - "docsUrl": "https://datahubproject.io/docs/metadata-ingestion/", - "recipe": "source:\n type: \n config:\n # Source-type specifics config\n " + "urn": "urn:li:dataPlatform:sigma", + "name": "sigma", + "displayName": "Sigma", + "description": "Import Workspaces, Workbooks, Pages, Elements, and lineage from Sigma Computing.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/sigma/", + "recipe": "source:\n type: sigma\n config:\n # Coordinates\n api_url: https://aws-api.sigmacomputing.com/v2\n # Coordinates\n client_id: CLIENT_ID\n client_secret: CLIENT_SECRET\n\n # Optional - filter for certain workspace names instead of ingesting everything.\n # workspace_pattern:\n\n # allow:\n # - workspace_name\n ingest_owner: true" }, { "urn": "urn:li:dataPlatform:qlik-sense", "name": "qlik-sense", "displayName": "Qlik Sense", + "description": "Import Spaces, Apps, Sheets, Charts, and Datasets from Qlik Sense.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/qlik-sense/", "recipe": "source:\n type: qlik-sense\n config:\n # Coordinates\n tenant_hostname: https://xyz12xz.us.qlikcloud.com\n # Coordinates\n api_key: QLIK_API_KEY\n\n # Optional - filter for certain space names instead of ingesting everything.\n # space_pattern:\n\n # allow:\n # - space_name\n ingest_owner: true" }, - { - "urn": "urn:li:dataPlatform:sigma", - "name": "sigma", - "displayName": "Sigma", - "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/sigma/", - "recipe": "source:\n type: sigma\n config:\n # Coordinates\n api_url: https://aws-api.sigmacomputing.com/v2\n # Coordinates\n client_id: CLIENT_ID\n client_secret: CLIENT_SECRET\n\n # Optional - filter for certain workspace names instead of ingesting everything.\n # workspace_pattern:\n\n # allow:\n # - workspace_name\n ingest_owner: true" - }, { "urn": "urn:li:dataPlatform:cockroachdb", "name": "cockroachdb", "displayName": "CockroachDb", + "description": "Import Databases, Schemas, Tables, Views, statistics and lineage from CockroachDB.", "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/cockroachdb/", "recipe": "source: \n type: cockroachdb\n config:\n # Coordinates\n host_port: # Your CockroachDb host and port, e.g. cockroachdb:5432\n database: # Your CockroachDb Database, e.g. sample_db\n\n # Credentials\n # Add secret in Secrets Tab with relevant names for each variable\n username: null # Your CockroachDb username, e.g. admin\n\n # Options\n include_tables: true\n include_views: true\n\n # Profiling\n profiling:\n enabled: true\n profile_table_level_only: true\n stateful_ingestion:\n enabled: true" + }, + { + "urn": "urn:li:dataPlatform:csv-enricher", + "name": "csv-enricher", + "displayName": "CSV", + "description": "Import metadata from a formatted CSV.", + "docsUrl": "https://datahubproject.io/docs/generated/ingestion/sources/csv'", + "recipe": "source: \n type: csv-enricher \n config: \n # URL of your csv file to ingest \n filename: \n array_delimiter: '|' \n delimiter: ',' \n write_semantics: PATCH" + }, + { + "urn": "urn:li:dataPlatform:custom", + "name": "custom", + "displayName": "Other", + "description": "Configure a custom recipe using YAML.", + "docsUrl": "https://datahubproject.io/docs/metadata-ingestion/", + "recipe": "source:\n type: \n config:\n # Source-type specifics config\n " } ] diff --git a/datahub-web-react/src/app/ingest/source/builder/types.ts b/datahub-web-react/src/app/ingest/source/builder/types.ts index 2df467b7beba1f..e42bd0b790b2c8 100644 --- a/datahub-web-react/src/app/ingest/source/builder/types.ts +++ b/datahub-web-react/src/app/ingest/source/builder/types.ts @@ -18,6 +18,7 @@ export interface SourceConfig { name: string; displayName: string; docsUrl: string; + description?: string; recipe: string; } diff --git a/datahub-web-react/src/app/ingest/source/executions/ExecutionRequestDetailsModal.tsx b/datahub-web-react/src/app/ingest/source/executions/ExecutionRequestDetailsModal.tsx index 0799f8af1173dc..6711f0ad12b03c 100644 --- a/datahub-web-react/src/app/ingest/source/executions/ExecutionRequestDetailsModal.tsx +++ b/datahub-web-react/src/app/ingest/source/executions/ExecutionRequestDetailsModal.tsx @@ -13,9 +13,13 @@ import { getExecutionRequestStatusDisplayText, getExecutionRequestStatusIcon, getExecutionRequestSummaryText, + getIngestionSourceStatus, + getStructuredReport, RUNNING, SUCCESS, } from '../utils'; +import { ExecutionRequestResult } from '../../../../types.generated'; +import { StructuredReport } from './reporting/StructuredReport'; const StyledTitle = styled(Typography.Title)` padding: 0px; @@ -125,26 +129,30 @@ export const ExecutionDetailsModal = ({ urn, visible, onClose }: Props) => { }; const logs = (showExpandedLogs && output) || output?.split('\n').slice(0, 5).join('\n'); - const result = data?.executionRequest?.result?.status; + const result = data?.executionRequest?.result as Partial; + const status = getIngestionSourceStatus(result); useEffect(() => { const interval = setInterval(() => { - if (result === RUNNING) refetch(); + if (status === RUNNING) refetch(); }, 2000); return () => clearInterval(interval); }); - const ResultIcon = result && getExecutionRequestStatusIcon(result); - const resultColor = result && getExecutionRequestStatusDisplayColor(result); - const resultText = result && ( + const ResultIcon = status && getExecutionRequestStatusIcon(status); + const resultColor = status && getExecutionRequestStatusDisplayColor(status); + const resultText = status && ( {ResultIcon && } - {getExecutionRequestStatusDisplayText(result)} + {getExecutionRequestStatusDisplayText(status)} ); + + const structuredReport = result && getStructuredReport(result); + const resultSummaryText = - (result && {getExecutionRequestSummaryText(result)}) || + (status && {getExecutionRequestSummaryText(status)}) || undefined; const recipeJson = data?.executionRequest?.input.arguments?.find((arg) => arg.key === 'recipe')?.value; @@ -167,21 +175,22 @@ export const ExecutionDetailsModal = ({ urn, visible, onClose }: Props) => { bodyStyle={modalBodyStyle} title={ - Ingestion Run Details + Sync Details } visible={visible} onCancel={onClose} > - {!data && loading && } - {error && message.error('Failed to load execution details :(')} + {!data && loading && } + {error && message.error('Failed to load sync details :(')}
Status {resultText} {resultSummaryText} + {structuredReport ? : null} - {result === SUCCESS && ( + {status === SUCCESS && ( {data?.executionRequest?.id && } @@ -190,7 +199,7 @@ export const ExecutionDetailsModal = ({ urn, visible, onClose }: Props) => { Logs - View logs that were collected during the ingestion run. + View logs that were collected during the sync.