Skip to content

Commit

Permalink
Visual polish
Browse files Browse the repository at this point in the history
  • Loading branch information
bengotow committed Mar 19, 2024
1 parent 10e8621 commit 3fb3117
Show file tree
Hide file tree
Showing 11 changed files with 94 additions and 45 deletions.
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import {Box, Caption, Colors, StyledTag, Tooltip} from '@dagster-io/ui-components';
import {Box, Caption, Colors, Icon, StyledTag, Tooltip} from '@dagster-io/ui-components';
import {Link} from 'react-router-dom';
import styled from 'styled-components';

Expand All @@ -9,18 +9,22 @@ import {
AssetNodeContainer,
} from './AssetNode';
import {AssetNodeFragment} from './types/AssetNode.types';
import {Timestamp} from '../app/time/Timestamp';
import {assetDetailsPathForKey} from '../assets/assetDetailsPathForKey';
import {AssetColumnLineageLocal} from '../assets/lineage/useColumnLineageDataForAssets';
import {AssetColumnLineageLocalColumn} from '../assets/lineage/useColumnLineageDataForAssets';
import {AssetComputeKindTag} from '../graph/OpTags';
import {AssetKeyInput} from '../graphql/types';
import {TypeTag} from '../metadata/TableSchema';
import {iconForColumnType} from '../metadata/TableSchema';

export const AssetColumnsGroupNode = ({
selected,
definition,
height,
asOf,
}: {
selected: boolean;
definition: AssetNodeFragment;
asOf: string | undefined;
height: number;
}) => {
return (
Expand All @@ -30,7 +34,15 @@ export const AssetColumnsGroupNode = ({
<AssetNodeBox $selected={selected} $isSource={definition.isSource} $noScale>
<AssetNameRow definition={definition} />
<Box style={{height: height - 60}} flex={{direction: 'column'}}></Box>
<Box border="top" padding={{horizontal: 8, vertical: 2}} style={{minHeight: 22}}>
{asOf ? (
<Caption color={Colors.textLighter()}>
<Timestamp timestamp={{ms: Number(asOf)}} />
</Caption>
) : undefined}
</Box>
</AssetNodeBox>
<AssetComputeKindTag definition={definition} style={{right: 10, paddingTop: 7}} />
</AssetNodeContainer>
</AssetInsetForHoverEffect>
);
Expand All @@ -42,20 +54,22 @@ export const AssetColumnNode = ({
selected,
}: {
assetKey: AssetKeyInput;
column: AssetColumnLineageLocal[''];
column: AssetColumnLineageLocalColumn;
selected: boolean;
}) => {
const icon = iconForColumnType(column.type ?? '');

return (
<Box margin={{horizontal: 12}} style={{height: 32}} flex={{direction: 'column'}}>
<Tooltip key={column.name} content={column.description || 'No description provided'}>
<ColumnLink
to={assetDetailsPathForKey(assetKey, {view: 'lineage', column: column.name})}
$selected={selected}
>
{icon ? <Icon name={icon} /> : <span style={{width: 16}} />}
<Caption style={{whiteSpace: 'nowrap', overflow: 'hidden', textOverflow: 'ellipsis'}}>
{column.name}
</Caption>
<TypeTag type={column.type || ''} />
</ColumnLink>
</Tooltip>
</Box>
Expand All @@ -65,11 +79,10 @@ export const AssetColumnNode = ({
const ColumnLink = styled(Link)<{$selected: boolean}>`
height: 28px;
margin: 2px 0;
padding-left: 8px;
padding-left: 2px;
padding-right: 4px;
display: flex;
gap: 4px;
justify-content: space-between;
align-items: center;
transition: background 100ms linear;
border-radius: 8px;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ import {
import {DagsterTypeSummary} from '../dagstertype/DagsterType';
import {DagsterTypeFragment} from '../dagstertype/types/DagsterType.types';
import {METADATA_ENTRY_FRAGMENT} from '../metadata/MetadataEntry';
import {TableSchemaLineageContext} from '../metadata/TableSchema';
import {Description} from '../pipelines/Description';
import {SidebarSection, SidebarTitle} from '../pipelines/SidebarComponents';
import {ResourceContainer, ResourceHeader} from '../pipelines/SidebarOpHelpers';
Expand Down Expand Up @@ -159,7 +160,12 @@ export const SidebarAssetInfo = ({graphNode}: {graphNode: GraphNode}) => {

{assetMetadata.length > 0 && (
<SidebarSection title="Metadata">
<AssetMetadataTable assetMetadata={assetMetadata} repoLocation={repoAddress?.location} />
<TableSchemaLineageContext.Provider value={{assetKey: asset.assetKey}}>
<AssetMetadataTable
assetMetadata={assetMetadata}
repoLocation={repoAddress?.location}
/>
</TableSchemaLineageContext.Provider>
</SidebarSection>
)}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,10 @@ export type LayoutAssetGraphConfig = dagre.GraphLabel & {
groupPaddingTop: number;
groupPaddingBottom: number;
groupRendering: 'if-varied' | 'always';

/** Supported in Dagre, just not documented. Additional spacing between group nodes */
clusterpaddingtop?: number;
clusterpaddingbottom?: number;
};

export type LayoutAssetGraphOptions = {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,9 @@ export const AssetColumnLineageGraph = ({
node: graphNode!,
};

const cols = columnLineageData[groupAssetGraphId] || {};
const colsAsOf = Object.values(cols)[0]?.asOf;

return (
<foreignObject
{...bounds}
Expand All @@ -92,6 +95,7 @@ export const AssetColumnLineageGraph = ({
definition={graphNode!.definition}
selected={focusedAssetGraphId === groupAssetGraphId}
height={bounds.height}
asOf={colsAsOf}
/>
</AssetNodeContextMenuWrapper>
</foreignObject>
Expand All @@ -117,6 +121,7 @@ export const AssetColumnLineageGraph = ({
description: 'Not found in column metadata',
type: null,
upstream: [],
asOf: undefined,
};

return (
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,11 @@ import {DagsterTypeSummary} from '../dagstertype/DagsterType';
import {AssetComputeKindTag} from '../graph/OpTags';
import {useStateWithStorage} from '../hooks/useStateWithStorage';
import {useLaunchPadHooks} from '../launchpad/LaunchpadHooksContext';
import {TableSchema, isCanonicalColumnLineageEntry} from '../metadata/TableSchema';
import {
TableSchema,
TableSchemaLineageContext,
isCanonicalColumnLineageEntry,
} from '../metadata/TableSchema';
import {RepositoryLink} from '../nav/RepositoryLink';
import {ScheduleOrSensorTag} from '../nav/ScheduleOrSensorTag';
import {useRepositoryLocationForAddress} from '../nav/useRepositoryLocationForAddress';
Expand Down Expand Up @@ -390,11 +394,14 @@ export const AssetNodeOverview = ({
</LargeCollapsibleSection>
{tableSchema && (
<LargeCollapsibleSection header="Columns" icon="view_column">
<TableSchema
schema={tableSchema.schema}
schemaLoadTimestamp={tableSchemaLoadTimestamp}
assetKeyForColumnSchema={columnSchema ? assetNode.assetKey : undefined}
/>
<TableSchemaLineageContext.Provider
value={{assetKey: columnSchema ? assetNode.assetKey : null}}
>
<TableSchema
schema={tableSchema.schema}
schemaLoadTimestamp={tableSchemaLoadTimestamp}
/>
</TableSchemaLineageContext.Provider>
</LargeCollapsibleSection>
)}
<LargeCollapsibleSection header="Metadata" icon="view_list">
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -366,7 +366,7 @@ function getQueryForVisibleAssets(
return {query: `+"${token}"+`, requestedDepth: 1};
}
if (view === 'lineage') {
const defaultDepth = lineageScope === 'neighbors' ? 2 : 5;
const defaultDepth = 1;
const requestedDepth = Number(lineageDepth) || defaultDepth;
const depthStr = '+'.repeat(requestedDepth);

Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
// eslint-disable-next-line no-restricted-imports

import {AssetNodeDefinitionFragment} from './types/AssetNodeDefinition.types';
import {AssetMaterializationFragment} from './types/useRecentAssetEvents.types';
import {AssetColumnLineageQuery} from './lineage/types/useColumnLineageDataForAssets.types';
import {isCanonicalColumnSchemaEntry} from '../metadata/TableSchema';

type AssetDefinitionWithMetadata = AssetColumnLineageQuery['assetNodes'][0];

/**
* This helper pulls the `columns` metadata entry from the most recent materialization
* and the asset definition, blending the two together to produce the most current
Expand All @@ -14,21 +15,20 @@ export function buildConsolidatedColumnSchema({
definition,
definitionLoadTimestamp,
}: {
materialization: Pick<AssetMaterializationFragment, 'metadataEntries' | 'timestamp'> | undefined;
definition: AssetNodeDefinitionFragment | undefined;
materialization:
| Pick<AssetDefinitionWithMetadata['assetMaterializations'][0], 'metadataEntries' | 'timestamp'>
| undefined;
definition: Pick<AssetDefinitionWithMetadata, 'metadataEntries'> | undefined;
definitionLoadTimestamp: number | undefined;
}) {
const materializationTableSchema = materialization?.metadataEntries.find(
isCanonicalColumnSchemaEntry,
);
const materializationTableSchemaLoadTimestamp = materialization
? Number(materialization.timestamp)
: undefined;
const materializationTimestamp = materialization ? Number(materialization.timestamp) : undefined;
const definitionTableSchema = definition?.metadataEntries.find(isCanonicalColumnSchemaEntry);

let tableSchema = materializationTableSchema ?? definitionTableSchema;
const tableSchemaLoadTimestamp =
materializationTableSchemaLoadTimestamp ?? definitionLoadTimestamp;
const tableSchemaLoadTimestamp = materializationTimestamp ?? definitionLoadTimestamp;

// Merge the descriptions from the definition table schema with the materialization table schema
if (materializationTableSchema && definitionTableSchema) {
Expand All @@ -46,6 +46,6 @@ export function buildConsolidatedColumnSchema({
schema: {...materializationTableSchema.schema, columns: mergedColumns},
};
}

console.log(tableSchema);
return {tableSchema, tableSchemaLoadTimestamp};
}
Original file line number Diff line number Diff line change
Expand Up @@ -19,16 +19,19 @@ export type AssetColumnLineageServer = {
}[];
};

export type AssetColumnLineageLocalColumn = {
name: string;
type: string | null;
description: string | null;
asOf: string | undefined; // materialization timestamp
upstream: {
assetKey: AssetKeyInput;
columnName: string;
}[];
};

export type AssetColumnLineageLocal = {
[column: string]: {
name: string;
type: string | null;
description: string | null;
upstream: {
assetKey: AssetKeyInput;
columnName: string;
}[];
};
[column: string]: AssetColumnLineageLocalColumn;
};

export type AssetColumnLineages = {[graphId: string]: AssetColumnLineageLocal | undefined};
Expand Down Expand Up @@ -64,6 +67,7 @@ const getColumnLineage = (
column,
{
name: column,
asOf: materialization?.timestamp,
type: schemaParsed[column]?.type || null,
description: schemaParsed[column]?.description || null,
upstream: m.map((u) => ({
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,8 @@ const LINEAGE_GRAPH_COLUMN_LAYOUT_OPTIONS: LayoutAssetGraphOptions = {
nodeHeight: 32,
nodesep: 0,
edgesep: 0,
groupPaddingBottom: -5,
clusterpaddingtop: 50,
groupPaddingBottom: -3,
groupPaddingTop: 68,
groupRendering: 'always',
},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,7 @@ export const MetadataEntry = ({
) : (
<MetadataEntryModalAction
label={entry.label}
modalWidth={900}
copyContent={() => JSON.stringify(entry.schema, null, 2)}
content={() => (
<Box
Expand Down Expand Up @@ -368,6 +369,7 @@ const PythonArtifactLink = ({
const MetadataEntryModalAction = (props: {
children: React.ReactNode;
label: string;
modalWidth?: number;
content: () => React.ReactNode;
copyContent: () => string;
}) => {
Expand All @@ -377,7 +379,7 @@ const MetadataEntryModalAction = (props: {
<MetadataEntryAction onClick={() => setOpen(true)}>{props.children}</MetadataEntryAction>
<Dialog
icon="info"
style={{width: 'auto', minWidth: 400, maxWidth: '80vw'}}
style={{width: props.modalWidth || 'auto', minWidth: 400, maxWidth: '80vw'}}
title={props.label}
onClose={() => setOpen(false)}
isOpen={open}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ import {
Tooltip,
} from '@dagster-io/ui-components';
import {Spacing} from '@dagster-io/ui-components/src/components/types';
import {useState} from 'react';
import {createContext, useContext, useState} from 'react';

import {TableSchemaFragment} from './types/TableSchema.types';
import {Timestamp} from '../app/time/Timestamp';
Expand All @@ -23,6 +23,7 @@ import {
MaterializationEvent,
TableSchemaMetadataEntry,
} from '../graphql/types';
import {Description} from '../pipelines/Description';
import {AnchorButton} from '../ui/AnchorButton';

type ITableSchema = TableSchemaFragment;
Expand All @@ -33,7 +34,6 @@ interface ITableSchemaProps {
schema: ITableSchema;
schemaLoadTimestamp?: number | undefined;
itemHorizontalPadding?: Spacing;
assetKeyForColumnSchema?: AssetKeyInput;
}

export const isCanonicalColumnSchemaEntry = (
Expand All @@ -45,12 +45,17 @@ export const isCanonicalColumnLineageEntry = (
m: Pick<MaterializationEvent['metadataEntries'][0], '__typename' | 'label'>,
): m is JsonMetadataEntry => m.__typename === 'JsonMetadataEntry' && m.label === 'lineage';

export const TableSchemaLineageContext = createContext<{assetKey: AssetKeyInput | null}>({
assetKey: null,
});

export const TableSchema = ({
schema,
schemaLoadTimestamp,
itemHorizontalPadding,
assetKeyForColumnSchema,
}: ITableSchemaProps) => {
const {assetKey} = useContext(TableSchemaLineageContext);

const multiColumnConstraints = schema.constraints?.other || [];
const [filter, setFilter] = useState('');
const rows = schema.columns.filter(
Expand Down Expand Up @@ -89,7 +94,7 @@ export const TableSchema = ({
<td>Column name</td>
<td style={{width: 200}}>Type</td>
<td>Description</td>
{assetKeyForColumnSchema ? <td style={{width: 56}} /> : undefined}
{assetKey ? <td style={{width: 56}} /> : undefined}
</tr>
</thead>
<tbody>
Expand All @@ -106,13 +111,15 @@ export const TableSchema = ({
<ArbitraryConstraintTag key={i} constraint={constraint} />
))}
</td>
<td>{column.description}</td>
{assetKeyForColumnSchema ? (
<td>
<Description description={column.description} />
</td>
{assetKey ? (
<td style={{padding: '4px 12px'}}>
<Tooltip content="View column lineage">
<AnchorButton
icon={<Icon name="column_lineage" />}
to={assetDetailsPathForKey(assetKeyForColumnSchema, {
to={assetDetailsPathForKey(assetKey, {
view: 'lineage',
column: column.name,
})}
Expand All @@ -135,7 +142,7 @@ export const TableSchema = ({
);
};

const iconForType = (type: string): IconName | null => {
export const iconForColumnType = (type: string): IconName | null => {
const lower = type.toLowerCase();
if (lower.includes('bool')) {
return 'datatype_bool';
Expand All @@ -161,7 +168,7 @@ export const TypeTag = ({type = ''}: {type: string}) => {
return <span />;
}

const icon = iconForType(type);
const icon = iconForColumnType(type);

return (
<Tag intent="none">
Expand Down

0 comments on commit 3fb3117

Please sign in to comment.