Skip to content

Commit

Permalink
fix: console showing subworkflows as unknown (#570)
Browse files Browse the repository at this point in the history
* fix: console showing subworkflows as unknown

Signed-off-by: Olga Nad <[email protected]>

* fix: replace comparison with lodash isEqual

Signed-off-by: Olga Nad <[email protected]>

* fix: remove checkIfObjectsAreSame

Signed-off-by: Olga Nad <[email protected]>

Signed-off-by: Olga Nad <[email protected]>
  • Loading branch information
olga-union authored Aug 19, 2022
1 parent a4bf213 commit cdd94b7
Show file tree
Hide file tree
Showing 5 changed files with 174 additions and 31 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
import * as React from 'react';
import * as moment from 'moment-timezone';
import makeStyles from '@material-ui/core/styles/makeStyles';
import { COLOR_SPECTRUM } from 'components/Theme/colorSpectrum';
import { useScaleContext } from './scaleContext';
import { TimeZone } from './helpers';

interface StyleProps {
chartWidth: number;
labelInterval: number;
}

const useStyles = makeStyles((_theme) => ({
chartHeader: (props: StyleProps) => ({
height: 41,
display: 'flex',
alignItems: 'center',
width: `${props.chartWidth}px`,
}),
taskDurationsLabelItem: (props: StyleProps) => ({
fontSize: 12,
fontFamily: 'Open Sans',
fontWeight: 'bold',
color: COLOR_SPECTRUM.gray40.color,
paddingLeft: 10,
width: `${props.labelInterval}px`,
}),
}));

interface HeaderProps extends StyleProps {
chartTimezone: string;
totalDurationSec: number;
startedAt: Date;
}

export const ChartHeader = (props: HeaderProps) => {
const styles = useStyles(props);

const { chartInterval: chartTimeInterval, setMaxValue } = useScaleContext();
const { startedAt, chartTimezone, totalDurationSec } = props;

React.useEffect(() => {
setMaxValue(props.totalDurationSec);
}, [props.totalDurationSec, setMaxValue]);

const labels = React.useMemo(() => {
const len = Math.ceil(totalDurationSec / chartTimeInterval);
const lbs = len > 0 ? new Array(len).fill('') : [];
return lbs.map((_, idx) => {
const time = moment.utc(new Date(startedAt.getTime() + idx * chartTimeInterval * 1000));
return chartTimezone === TimeZone.UTC
? time.format('hh:mm:ss A')
: time.local().format('hh:mm:ss A');
});
}, [chartTimezone, startedAt, chartTimeInterval, totalDurationSec]);

return (
<div className={styles.chartHeader}>
{labels.map((label) => {
return (
<div className={styles.taskDurationsLabelItem} key={label}>
{label}
</div>
);
})}
</div>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,9 @@ import { createRef, useContext, useEffect, useRef, useState } from 'react';
import { NodeExecutionsByIdContext } from 'components/Executions/contexts';
import { checkForDynamicExecutions } from 'components/common/utils';
import { convertToPlainNodes } from './helpers';
import { ChartHeader } from './chartHeader';
import { ChartHeader } from './ChartHeader';
import { useScaleContext } from './scaleContext';
import { TaskNames } from './taskNames';
import { TaskNames } from './TaskNames';
import { getChartDurationData } from './TimelineChart/chartData';
import { TimelineChart } from './TimelineChart';

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
import * as React from 'react';
import { makeStyles, Theme, Typography } from '@material-ui/core';

import { RowExpander } from 'components/Executions/Tables/RowExpander';
import { getNodeTemplateName } from 'components/WorkflowGraph/utils';
import { dNode } from 'models/Graph/types';
import { NodeExecutionName } from './NodeExecutionName';
import { NodeExecutionsTimelineContext } from './context';

const useStyles = makeStyles((theme: Theme) => ({
taskNamesList: {
overflowY: 'scroll',
flex: 1,
},
namesContainer: {
display: 'flex',
flexDirection: 'row',
alignItems: 'flex-start',
justifyContent: 'left',
padding: '0 10px',
height: 56,
width: 256,
borderBottom: `1px solid ${theme.palette.divider}`,
whiteSpace: 'nowrap',
},
namesContainerExpander: {
display: 'flex',
marginTop: 'auto',
marginBottom: 'auto',
},
namesContainerBody: {
display: 'flex',
flexDirection: 'column',
alignItems: 'flex-start',
justifyContent: 'center',
whiteSpace: 'nowrap',
height: '100%',
overflow: 'hidden',
},
displayName: {
marginTop: 4,
textOverflow: 'ellipsis',
width: '100%',
overflow: 'hidden',
},
leaf: {
width: 30,
},
}));

interface TaskNamesProps {
nodes: dNode[];
onScroll: () => void;
onToggle: (id: string, scopeId: string, level: number) => void;
}

export const TaskNames = React.forwardRef<HTMLDivElement, TaskNamesProps>((props, ref) => {
const state = React.useContext(NodeExecutionsTimelineContext);
const { nodes, onScroll, onToggle } = props;
const styles = useStyles();

return (
<div className={styles.taskNamesList} ref={ref} onScroll={onScroll}>
{nodes.map((node) => {
const templateName = getNodeTemplateName(node);
const nodeLevel = node?.level ?? 0;
return (
<div
className={styles.namesContainer}
key={`level=${nodeLevel}-id=${node.id}-name=${node.scopedId}`}
style={{ paddingLeft: nodeLevel * 16 }}
>
<div className={styles.namesContainerExpander}>
{node.nodes?.length ? (
<RowExpander
expanded={node.expanded || false}
onClick={() => onToggle(node.id, node.scopedId, nodeLevel)}
/>
) : (
<div className={styles.leaf} />
)}
</div>

<div className={styles.namesContainerBody}>
<NodeExecutionName
name={node.name}
execution={node.execution!} // some nodes don't have associated execution
state={state}
/>
<Typography variant="subtitle1" color="textSecondary" className={styles.displayName}>
{templateName}
</Typography>
</div>
</div>
);
})}
</div>
);
});
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,7 @@ import { formatDateLocalTimezone, formatDateUTC, millisecondsToHMS } from 'commo
import { timestampToDate } from 'common/utils';
import { useCommonStyles } from 'components/common/styles';
import { isEqual } from 'lodash';
import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums';
import { TaskNodeMetadata } from 'models/Execution/types';
import { NodeExecutionPhase } from 'models/Execution/enums';
import * as React from 'react';
import { useNodeExecutionContext } from '../contextProvider/NodeExecutionDetails';
import { ExecutionStatusBadge } from '../ExecutionStatusBadge';
Expand Down Expand Up @@ -107,15 +106,6 @@ const DisplayType: React.FC<NodeExecutionCellRendererData> = ({ execution }) =>
return <Typography color="textSecondary">{type}</Typography>;
};

const hiddenCacheStatuses = [CatalogCacheStatus.CACHE_MISS, CatalogCacheStatus.CACHE_DISABLED];
function hasCacheStatus(taskNodeMetadata?: TaskNodeMetadata): taskNodeMetadata is TaskNodeMetadata {
if (!taskNodeMetadata) {
return false;
}
const { cacheStatus } = taskNodeMetadata;
return !hiddenCacheStatuses.includes(cacheStatus);
}

export function generateColumns(
styles: ReturnType<typeof useColumnStyles>,
): NodeExecutionColumnDefinition[] {
Expand Down
22 changes: 4 additions & 18 deletions packages/zapp/console/src/components/WorkflowGraph/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,22 +27,6 @@ export function isExpanded(node: any) {
return !!node.expanded;
}

/**
* Utility funciton assumes (loose) parity between [a]->[b] if matching
* keys have matching values.
* @param a object
* @param b object
* @returns boolean
*/
export const checkIfObjectsAreSame = (a, b) => {
for (const k in a) {
if (a[k] != b[k]) {
return false;
}
}
return true;
};

/**
* Returns a display name from either workflows or nodes
* @param context input can be either CompiledWorkflow or CompiledNode
Expand Down Expand Up @@ -110,23 +94,25 @@ export const getNodeTypeFromCompiledNode = (node: CompiledNode): dTypes => {
};

export const getSubWorkflowFromId = (id, workflow) => {
const _ = require('lodash');
const { subWorkflows } = workflow;
/* Find current matching entitity from subWorkflows */
for (const k in subWorkflows) {
const subWorkflowId = subWorkflows[k].template.id;
if (checkIfObjectsAreSame(subWorkflowId, id)) {
if (_.isEqual(subWorkflowId, id)) {
return subWorkflows[k];
}
}
return false;
};

export const getTaskTypeFromCompiledNode = (taskNode: TaskNode, tasks: CompiledTask[]) => {
const _ = require('lodash');
for (let i = 0; i < tasks.length; i++) {
const compiledTask: CompiledTask = tasks[i];
const taskTemplate: TaskTemplate = compiledTask.template;
const templateId: Identifier = taskTemplate.id;
if (checkIfObjectsAreSame(templateId, taskNode.referenceId)) {
if (_.isEqual(templateId, taskNode.referenceId)) {
return compiledTask;
}
}
Expand Down

0 comments on commit cdd94b7

Please sign in to comment.