From ad6549ad2e9455d79a496cfd77d8d15ce7d2dc0e Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Mon, 28 Mar 2022 10:55:49 -0700 Subject: [PATCH 1/6] fix: add cached info to timeline Signed-off-by: Nastya Rusina --- .storybook/preview.js | 1 + .../Timeline/BarChart/chartData.tsx | 59 +++++++ .../Timeline/BarChart/utils.tsx | 112 +++++++++++++ .../Timeline/ExecutionTimeline.tsx | 61 ++++--- .../Timeline/Timeline.stories.tsx | 156 ++++++++++++++++++ .../ExecutionDetails/Timeline/chartData.tsx | 129 --------------- .../test/NodeExecutionDetails.test.tsx | 23 +-- .../Executions/NodeExecutionCacheStatus.tsx | 18 +- .../Tables/nodeExecutionColumns.tsx | 8 +- .../Tables/test/NodeExecutionsTable.test.tsx | 22 +-- src/components/Executions/constants.ts | 17 +- .../Executions/nodeExecutionQueries.ts | 2 +- src/models/Execution/enums.ts | 2 + 13 files changed, 408 insertions(+), 202 deletions(-) create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx create mode 100644 src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx delete mode 100644 src/components/Executions/ExecutionDetails/Timeline/chartData.tsx diff --git a/.storybook/preview.js b/.storybook/preview.js index 1b2809c1a..776dc9908 100644 --- a/.storybook/preview.js +++ b/.storybook/preview.js @@ -1,6 +1,7 @@ import React from 'react'; import { StorybookContainer } from './StorybookContainer'; +//👇 Configures Storybook to log 'onXxx' actions (example: onArchiveTask and onPinTask ) in the UI export const parameters = { actions: { argTypesRegex: '^on[A-Z].*' }, controls: { diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx new file mode 100644 index 000000000..801471f45 --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx @@ -0,0 +1,59 @@ +import { timestampToDate } from 'common/utils'; +import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums'; +import { dNode } from 'models/Graph/types'; +import { BarItemData } from './utils'; + +const EMPTY_BAR_ITEM: BarItemData = { + phase: NodeExecutionPhase.UNDEFINED, + startOffsetSec: 0, + durationSec: 0, + isFromCache: false, +}; + +export const getChartDurationData = (nodes: dNode[], startedAt: Date): BarItemData[] => { + if (nodes.length === 0) return []; + + const initialStartTime = startedAt.getTime(); + const result: BarItemData[] = nodes.map(({ execution }) => { + if (!execution) { + return EMPTY_BAR_ITEM; + } + + const phase = execution.closure.phase; + const isFromCache = + execution.closure.taskNodeMetadata?.cacheStatus === CatalogCacheStatus.CACHE_HIT; + + // Offset values + let startOffset = 0; + const startedAt = execution.closure.startedAt; + if (isFromCache) { + if (execution.closure.createdAt) { + startOffset = timestampToDate(execution.closure.createdAt).getTime() - initialStartTime; + } + } else if (startedAt) { + startOffset = timestampToDate(startedAt).getTime() - initialStartTime; + } + + // duration + let durationSec = 0; + if (isFromCache) { + const updatedAt = execution.closure.updatedAt?.seconds?.toNumber() ?? 0; + const createdAt = execution.closure.createdAt?.seconds?.toNumber() ?? 0; + durationSec = updatedAt - createdAt; + durationSec = durationSec === 0 ? 2 : durationSec; + } else if (phase === NodeExecutionPhase.RUNNING) { + // we need to add check if parents are failed - workflow status - ? + if (startedAt) { + const duration = Date.now() - timestampToDate(startedAt).getTime(); + durationSec = duration / 1000; + } + } else { + durationSec = execution.closure.duration?.seconds?.toNumber() ?? 0; + } + + return { phase, startOffsetSec: startOffset / 1000, durationSec, isFromCache }; + }); + + // Do we want to get initialStartTime from different place, to avoid recalculating it. + return result; +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx new file mode 100644 index 000000000..3a48c637d --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx @@ -0,0 +1,112 @@ +import { getNodeExecutionPhaseConstants } from 'components/Executions/utils'; +import { primaryTextColor } from 'components/Theme/constants'; +import { NodeExecutionPhase } from 'models/Execution/enums'; + +const CASHED_GREEN = 'rgba(74,227,174,0.25)'; // statusColors.SUCCESS (Mint20) with 25% opacity +const TRANSPARENT = 'rgba(0, 0, 0, 0)'; + +export interface BarItemData { + phase: NodeExecutionPhase; + startOffsetSec: number; + durationSec: number; + isFromCache: boolean; +} + +interface ChartDataInput { + elementsNumber: number; + totalDurationSec: number; + durations: number[]; + startOffset: number[]; + offsetColor: string[]; + barLabel: string[]; + barColor: string[]; +} + +// narusina - check if exports are still needed +export const getOffsetColor = (isCachedValue: boolean[]) => { + const colors = isCachedValue.map((val) => (val === true ? CASHED_GREEN : TRANSPARENT)); + return colors; +}; + +/** + * Generates chart data maps per each BarItemData ("node") section + */ +export const generateChartData = (data: BarItemData[]): ChartDataInput => { + const durations: number[] = []; + const startOffset: number[] = []; + const offsetColor: string[] = []; + const barLabel: string[] = []; + const barColor: string[] = []; + + let totalDurationSec = 0; + data.forEach((element) => { + durations.push(element.durationSec); + startOffset.push(element.startOffsetSec); + offsetColor.push(element.isFromCache ? CASHED_GREEN : TRANSPARENT); + barLabel.push(element.isFromCache ? 'From cache' : `${Math.round(element.durationSec)}s`); + barColor.push( + getNodeExecutionPhaseConstants(element.phase ?? NodeExecutionPhase.UNDEFINED).badgeColor, + ); + + totalDurationSec = Math.max(totalDurationSec, element.startOffsetSec + element.durationSec); + }); + const elementsNumber = data.length; + return { + elementsNumber, + totalDurationSec, + durations, + startOffset, + offsetColor, + barLabel, + barColor, + }; +}; + +/** + * Generates chart data format suitable for Chart.js Bar. Each bar consists of two data items: + * |-----------|XXXXXXXXXXXXXXXX| + * |-|XXXXXX| + * |------|XXXXXXXXXXXXX| + * Where |---| is offset - usually transparent part to give user a feeling that timeline wasn't started from ZERO time position + * Where |XXX| is duration of the operation, colored per step Phase status. + */ +export const getChartData = (data: ChartDataInput) => { + const defaultStyle = { + barPercentage: 1, + borderWidth: 0, + }; + + return { + labels: Array(data.elementsNumber).fill(''), // clear up Chart Bar default labels + datasets: [ + // fill-in offsets + { + ...defaultStyle, + data: data.startOffset, + backgroundColor: data.offsetColor, + datalabels: { + labels: { + title: null, + }, + }, + }, + // fill in duration bars + { + ...defaultStyle, + data: data.durations, + backgroundColor: data.barColor, + datalabels: { + // Positioning info - https://chartjs-plugin-datalabels.netlify.app/guide/positioning.html + color: primaryTextColor, + align: 'end' as const, // related to text + anchor: 'start' as const, // related to bar + formatter: function (value, context) { + return data.offsetColor[context.dataIndex] === CASHED_GREEN + ? '\u229A From Cache' + : `${Math.round(value)}s`; + }, + }, + }, + ], + }; +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx index 3430e32b3..a8538a5c1 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx @@ -9,20 +9,22 @@ import { useNodeExecutionContext } from 'components/Executions/contextProvider/N import { transformerWorkflowToDag } from 'components/WorkflowGraph/transformerWorkflowToDag'; import { isEndNode, isStartNode, isExpanded } from 'components/WorkflowGraph/utils'; import { tableHeaderColor } from 'components/Theme/constants'; +import { timestampToDate } from 'common/utils'; import { NodeExecution } from 'models/Execution/types'; import { dNode } from 'models/Graph/types'; -import { TaskNames } from './taskNames'; +import { generateChartData, getChartData } from './BarChart/utils'; +import { getChartDurationData } from './BarChart/chartData'; import { convertToPlainNodes, getBarOptions, TimeZone } from './helpers'; import { ChartHeader } from './chartHeader'; -import { useChartDurationData } from './chartData'; import { useScaleContext } from './scaleContext'; +import { TaskNames } from './taskNames'; // Register components to be usable by chart.js ChartJS.register(...registerables, ChartDataLabels); interface StyleProps { chartWidth: number; - durationLength: number; + itemsShown: number; } const useStyles = makeStyles((theme) => ({ @@ -30,7 +32,7 @@ const useStyles = makeStyles((theme) => ({ marginTop: -10, marginLeft: -15, width: `${props.chartWidth + 20}px`, - height: `${56 * props.durationLength + 20}px`, + height: `${56 * props.itemsShown + 20}px`, }), taskNames: { display: 'flex', @@ -85,6 +87,7 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime const [originalNodes, setOriginalNodes] = React.useState([]); const [showNodes, setShowNodes] = React.useState([]); + const [startedAt, setStartedAt] = React.useState(new Date()); const { compiledWorkflowClosure } = useNodeExecutionContext(); @@ -99,39 +102,45 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime React.useEffect(() => { const initializeNodes = convertToPlainNodes(originalNodes); - setShowNodes( - initializeNodes.map((node) => { - const index = nodeExecutions.findIndex((exe) => exe.scopedId === node.scopedId); - return { - ...node, - execution: index >= 0 ? nodeExecutions[index] : undefined, - }; - }), - ); + const updatedShownNodesMap = initializeNodes.map((node) => { + const index = nodeExecutions.findIndex((exe) => exe.scopedId === node.scopedId); + return { + ...node, + execution: index >= 0 ? nodeExecutions[index] : undefined, + }; + }); + setShowNodes(updatedShownNodesMap); + + // set startTime for all timeline offset and duration calculations. + const firstStartedAt = updatedShownNodesMap[0]?.execution?.closure.startedAt; + if (firstStartedAt) { + setStartedAt(timestampToDate(firstStartedAt)); + } }, [originalNodes, nodeExecutions]); - const { startedAt, totalDuration, durationLength, chartData } = useChartDurationData({ - nodes: showNodes, - }); + const barItemsData = getChartDurationData(showNodes, startedAt); + const chartDataInput = generateChartData(barItemsData); const { chartInterval: chartTimeInterval, setMaxValue } = useScaleContext(); - const styles = useStyles({ chartWidth: chartWidth, durationLength: durationLength }); + const styles = useStyles({ chartWidth: chartWidth, itemsShown: chartDataInput.elementsNumber }); React.useEffect(() => { - setMaxValue(totalDuration); - }, [totalDuration, setMaxValue]); + setMaxValue(chartDataInput.totalDurationSec); + }, [chartDataInput.totalDurationSec, setMaxValue]); React.useEffect(() => { - const calcWidth = Math.ceil(totalDuration / chartTimeInterval) * INTERVAL_LENGTH; + const calcWidth = + Math.ceil(chartDataInput.totalDurationSec / chartTimeInterval) * INTERVAL_LENGTH; if (durationsRef.current && calcWidth < durationsRef.current.clientWidth) { setLabelInterval( - durationsRef.current.clientWidth / Math.ceil(totalDuration / chartTimeInterval), + durationsRef.current.clientWidth / + Math.ceil(chartDataInput.totalDurationSec / chartTimeInterval), ); setChartWidth(durationsRef.current.clientWidth); } else { setChartWidth(calcWidth); setLabelInterval(INTERVAL_LENGTH); } - }, [totalDuration, chartTimeInterval, durationsRef]); + }, [chartDataInput.totalDurationSec, chartTimeInterval, durationsRef]); const onGraphScroll = () => { // cover horizontal scroll only @@ -178,15 +187,15 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime }; const labels = React.useMemo(() => { - const len = Math.ceil(totalDuration / chartTimeInterval); - const lbs = len > 0 ? new Array(len).fill(0) : []; + const len = Math.ceil(chartDataInput.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, totalDuration]); + }, [chartTimezone, startedAt, chartTimeInterval, chartDataInput.totalDurationSec]); return ( <> @@ -205,7 +214,7 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime
- +
diff --git a/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx b/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx new file mode 100644 index 000000000..173697dbe --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx @@ -0,0 +1,156 @@ +import { ComponentMeta, ComponentStory } from '@storybook/react'; +import * as React from 'react'; +import { Bar } from 'react-chartjs-2'; +import { Chart as ChartJS, registerables } from 'chart.js'; +import ChartDataLabels from 'chartjs-plugin-datalabels'; +import { statusColors } from 'components/Theme/constants'; +import { NodeExecutionPhase } from 'models/Execution/enums'; +import { getNodeExecutionPhaseConstants } from 'components/Executions/utils'; +import { generateChartData, getChartData, getOffsetColor } from './BarChart/utils'; + +ChartJS.register(...registerables, ChartDataLabels); + +const mockData = [5, 2, 10, 25, 20]; +const startOffset = [0, 10, 0, 15, 7]; +const isCachedValue = [false, true, false, false, false]; +const backgroundColor = [ + statusColors.FAILURE, + statusColors.QUEUED, + statusColors.RUNNING, + statusColors.WARNING, + statusColors.SUCCESS, +]; + +const isFromCache = [false, true, true, false, false, false]; +const phaseStatus = [ + NodeExecutionPhase.FAILED, + NodeExecutionPhase.SUCCEEDED, + NodeExecutionPhase.SUCCEEDED, + NodeExecutionPhase.RUNNING, + NodeExecutionPhase.UNDEFINED, + NodeExecutionPhase.SUCCEEDED, +]; + +const barItems = [ + { phase: phaseStatus[0], startOffsetSec: 0, durationSec: 5, isFromCache: isFromCache[0] }, + { phase: phaseStatus[1], startOffsetSec: 10, durationSec: 2, isFromCache: isFromCache[1] }, + { phase: phaseStatus[2], startOffsetSec: 0, durationSec: 1, isFromCache: isFromCache[2] }, + { phase: phaseStatus[3], startOffsetSec: 0, durationSec: 10, isFromCache: isFromCache[3] }, + { phase: phaseStatus[4], startOffsetSec: 15, durationSec: 25, isFromCache: isFromCache[4] }, + { phase: phaseStatus[5], startOffsetSec: 7, durationSec: 20, isFromCache: isFromCache[5] }, +]; + +const chartData = { + labels: mockData.map(() => ''), + datasets: [ + { + data: startOffset, + backgroundColor: getOffsetColor(isCachedValue), + barPercentage: 1, + borderWidth: 0, + datalabels: { + labels: { + title: null, + }, + }, + }, + { + data: mockData.map((duration) => { + return duration === -1 ? 10 : duration === 0 ? 0.5 : duration; + }), + backgroundColor: backgroundColor, + barPercentage: 1, + borderWidth: 0, + datalabels: { + color: '#292936' as const, + align: 'end' as const, + anchor: 'start' as const, + formatter: function (value, context) { + if (mockData[context.dataIndex] === -1) { + return ''; + } + return Math.round(value) + 's'; + }, + }, + }, + ], +}; + +const getBarOptions = ( + chartTimeInterval: number, + isFromCache: boolean[], + phaseStatus: NodeExecutionPhase[], +) => { + return { + animation: false as const, + indexAxis: 'y' as const, + elements: { + bar: { + borderWidth: 2, + }, + }, + responsive: true, + maintainAspectRatio: false, + plugins: { + legend: { + display: false, + }, + title: { + display: false, + }, + tooltip: { + filter: function (tooltipItem) { + // no tooltip for offsets + return tooltipItem.datasetIndex === 1; + }, + // Setting up tooltip: https://www.chartjs.org/docs/latest/configuration/tooltip.html + callbacks: { + label: function (context) { + const index = context.dataIndex; + let label = getNodeExecutionPhaseConstants(phaseStatus[index]).text; + if (context.parsed.x !== null) { + label += `: ${context.parsed.x}s`; + } + + if (isFromCache[index]) { + return [label, 'Read from cache']; + } + return label; + }, + }, + }, + }, + scales: { + x: { + format: Intl.DateTimeFormat, + position: 'top' as const, + ticks: { + display: false, + autoSkip: false, + stepSize: chartTimeInterval, + }, + stacked: true, + }, + y: { + stacked: true, + }, + }, + }; +}; + +export default { + title: 'Workflow/Timeline', + component: Bar, +} as ComponentMeta; + +const Template: ComponentStory = (args) => ; +export const SingleBar = Template.bind({}); +SingleBar.args = { + options: getBarOptions(1, isFromCache, phaseStatus), + data: getChartData(generateChartData(barItems)), +}; + +export const BarSection = () => { + // Chart interval - 1s + return ; +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/chartData.tsx b/src/components/Executions/ExecutionDetails/Timeline/chartData.tsx deleted file mode 100644 index dbef7d932..000000000 --- a/src/components/Executions/ExecutionDetails/Timeline/chartData.tsx +++ /dev/null @@ -1,129 +0,0 @@ -import { durationToMilliseconds, timestampToDate } from 'common/utils'; -import { getNodeExecutionPhaseConstants } from 'components/Executions/utils'; -import { NodeExecutionPhase } from 'models/Execution/enums'; -import { dNode } from 'models/Graph/types'; -import * as React from 'react'; - -interface DataProps { - nodes: dNode[]; -} - -export const useChartDurationData = (props: DataProps) => { - const colorData = React.useMemo(() => { - const definedExecutions = props.nodes.map( - ({ execution }) => - getNodeExecutionPhaseConstants(execution?.closure.phase ?? NodeExecutionPhase.UNDEFINED) - .badgeColor, - ); - return definedExecutions; - }, [props.nodes]); - - const startedAt = React.useMemo(() => { - if (props.nodes.length === 0 || !props.nodes[0].execution?.closure.startedAt) { - return new Date(); - } - return timestampToDate(props.nodes[0].execution?.closure.startedAt); - }, [props.nodes]); - - const stackedData = React.useMemo(() => { - let undefinedStart = 0; - for (const node of props.nodes) { - const exec = node.execution; - if (exec?.closure.startedAt) { - const startedTime = timestampToDate(exec?.closure.startedAt).getTime(); - const absoluteDuration = - startedTime - - startedAt.getTime() + - (exec?.closure.duration - ? durationToMilliseconds(exec?.closure.duration) - : Date.now() - startedTime); - if (absoluteDuration > undefinedStart) { - undefinedStart = absoluteDuration; - } - } - } - undefinedStart = undefinedStart / 1000; - - const definedExecutions = props.nodes.map(({ execution }) => - execution?.closure.startedAt - ? (timestampToDate(execution?.closure.startedAt).getTime() - startedAt.getTime()) / 1000 - : 0, - ); - - return definedExecutions; - }, [props.nodes, startedAt]); - - // Divide by 1000 to calculate all duration data be second based. - const durationData = React.useMemo(() => { - const definedExecutions = props.nodes.map((node) => { - const exec = node.execution; - if (!exec) return 0; - if (exec.closure.phase === NodeExecutionPhase.RUNNING) { - if (!exec.closure.startedAt) { - return 0; - } - return (Date.now() - timestampToDate(exec.closure.startedAt).getTime()) / 1000; - } - if (!exec.closure.duration) { - return 0; - } - return durationToMilliseconds(exec.closure.duration) / 1000; - }); - return definedExecutions; - }, [props.nodes]); - - const totalDuration = React.useMemo(() => { - const durations = durationData.map((duration, idx) => duration + stackedData[idx]); - return Math.max(...durations); - }, [durationData, stackedData]); - - const stackedColorData = React.useMemo(() => { - return durationData.map((duration) => { - return duration === 0 ? '#4AE3AE40' : 'rgba(0, 0, 0, 0)'; - }); - }, [durationData]); - - const chartData = React.useMemo(() => { - return { - labels: durationData.map(() => ''), - datasets: [ - { - data: stackedData, - backgroundColor: stackedColorData, - barPercentage: 1, - borderWidth: 0, - datalabels: { - labels: { - title: null, - }, - }, - }, - { - data: durationData.map((duration) => { - return duration === -1 ? 10 : duration === 0 ? 0.5 : duration; - }), - backgroundColor: colorData, - barPercentage: 1, - borderWidth: 0, - datalabels: { - color: '#292936' as const, - align: 'start' as const, - formatter: function (value, context) { - if (durationData[context.dataIndex] === -1) { - return ''; - } - return Math.round(value) + 's'; - }, - }, - }, - ], - }; - }, [durationData, stackedData, colorData, stackedColorData]); - - return { - startedAt, - totalDuration, - durationLength: durationData.length, - chartData, - }; -}; diff --git a/src/components/Executions/ExecutionDetails/test/NodeExecutionDetails.test.tsx b/src/components/Executions/ExecutionDetails/test/NodeExecutionDetails.test.tsx index 74bf66482..02570b12a 100644 --- a/src/components/Executions/ExecutionDetails/test/NodeExecutionDetails.test.tsx +++ b/src/components/Executions/ExecutionDetails/test/NodeExecutionDetails.test.tsx @@ -1,11 +1,12 @@ import { render, waitFor } from '@testing-library/react'; import { cacheStatusMessages, viewSourceExecutionString } from 'components/Executions/constants'; import { NodeExecutionDetailsContextProvider } from 'components/Executions/contextProvider/NodeExecutionDetails'; -import { Core } from 'flyteidl'; import { basicPythonWorkflow } from 'mocks/data/fixtures/basicPythonWorkflow'; import { mockWorkflowId } from 'mocks/data/fixtures/types'; import { insertFixture } from 'mocks/data/insertFixture'; import { mockServer } from 'mocks/server'; +import { ResourceType } from 'models/Common/types'; +import { CatalogCacheStatus } from 'models/Execution/enums'; import { NodeExecution, TaskNodeMetadata } from 'models/Execution/types'; import { mockExecution as mockTaskExecution } from 'models/Execution/__mocks__/mockTaskExecutionsData'; import * as React from 'react'; @@ -53,10 +54,10 @@ describe('NodeExecutionDetails', () => { let taskNodeMetadata: TaskNodeMetadata; beforeEach(() => { taskNodeMetadata = { - cacheStatus: Core.CatalogCacheStatus.CACHE_MISS, + cacheStatus: CatalogCacheStatus.CACHE_MISS, catalogKey: { datasetId: makeIdentifier({ - resourceType: Core.ResourceType.DATASET, + resourceType: ResourceType.DATASET, }), sourceTaskExecution: { ...mockTaskExecution.id }, }, @@ -66,14 +67,14 @@ describe('NodeExecutionDetails', () => { }); [ - Core.CatalogCacheStatus.CACHE_DISABLED, - Core.CatalogCacheStatus.CACHE_HIT, - Core.CatalogCacheStatus.CACHE_LOOKUP_FAILURE, - Core.CatalogCacheStatus.CACHE_MISS, - Core.CatalogCacheStatus.CACHE_POPULATED, - Core.CatalogCacheStatus.CACHE_PUT_FAILURE, + CatalogCacheStatus.CACHE_DISABLED, + CatalogCacheStatus.CACHE_HIT, + CatalogCacheStatus.CACHE_LOOKUP_FAILURE, + CatalogCacheStatus.CACHE_MISS, + CatalogCacheStatus.CACHE_POPULATED, + CatalogCacheStatus.CACHE_PUT_FAILURE, ].forEach((cacheStatusValue) => - it(`renders correct status for ${Core.CatalogCacheStatus[cacheStatusValue]}`, async () => { + it(`renders correct status for ${CatalogCacheStatus[cacheStatusValue]}`, async () => { taskNodeMetadata.cacheStatus = cacheStatusValue; mockServer.insertNodeExecution(execution); const { getByText } = renderComponent(); @@ -82,7 +83,7 @@ describe('NodeExecutionDetails', () => { ); it('renders source execution link for cache hits', async () => { - taskNodeMetadata.cacheStatus = Core.CatalogCacheStatus.CACHE_HIT; + taskNodeMetadata.cacheStatus = CatalogCacheStatus.CACHE_HIT; const sourceWorkflowExecutionId = taskNodeMetadata.catalogKey!.sourceTaskExecution.nodeExecutionId.executionId; const { getByText } = renderComponent(); diff --git a/src/components/Executions/NodeExecutionCacheStatus.tsx b/src/components/Executions/NodeExecutionCacheStatus.tsx index 6d3a10558..72b1fbf14 100644 --- a/src/components/Executions/NodeExecutionCacheStatus.tsx +++ b/src/components/Executions/NodeExecutionCacheStatus.tsx @@ -7,7 +7,7 @@ import classnames from 'classnames'; import { assertNever } from 'common/utils'; import { PublishedWithChangesOutlined } from 'components/common/PublishedWithChanges'; import { useCommonStyles } from 'components/common/styles'; -import { Core } from 'flyteidl'; +import { CatalogCacheStatus } from 'models/Execution/enums'; import { TaskNodeMetadata } from 'models/Execution/types'; import * as React from 'react'; import { Link as RouterLink } from 'react-router-dom'; @@ -29,25 +29,25 @@ const useStyles = makeStyles((theme: Theme) => ({ }, })); -/** Renders the appropriate icon for a given `Core.CatalogCacheStatus` */ +/** Renders the appropriate icon for a given CatalogCacheStatus */ export const NodeExecutionCacheStatusIcon: React.FC< SvgIconProps & { - status: Core.CatalogCacheStatus; + status: CatalogCacheStatus; } > = React.forwardRef(({ status, ...props }, ref) => { switch (status) { - case Core.CatalogCacheStatus.CACHE_DISABLED: - case Core.CatalogCacheStatus.CACHE_MISS: { + case CatalogCacheStatus.CACHE_DISABLED: + case CatalogCacheStatus.CACHE_MISS: { return ; } - case Core.CatalogCacheStatus.CACHE_HIT: { + case CatalogCacheStatus.CACHE_HIT: { return ; } - case Core.CatalogCacheStatus.CACHE_POPULATED: { + case CatalogCacheStatus.CACHE_POPULATED: { return ; } - case Core.CatalogCacheStatus.CACHE_LOOKUP_FAILURE: - case Core.CatalogCacheStatus.CACHE_PUT_FAILURE: { + case CatalogCacheStatus.CACHE_LOOKUP_FAILURE: + case CatalogCacheStatus.CACHE_PUT_FAILURE: { return ; } default: { diff --git a/src/components/Executions/Tables/nodeExecutionColumns.tsx b/src/components/Executions/Tables/nodeExecutionColumns.tsx index adac6b93f..e03d00d47 100644 --- a/src/components/Executions/Tables/nodeExecutionColumns.tsx +++ b/src/components/Executions/Tables/nodeExecutionColumns.tsx @@ -2,9 +2,8 @@ import { Tooltip, Typography } from '@material-ui/core'; import { formatDateLocalTimezone, formatDateUTC, millisecondsToHMS } from 'common/formatters'; import { timestampToDate } from 'common/utils'; import { useCommonStyles } from 'components/common/styles'; -import { Core } from 'flyteidl'; import { isEqual } from 'lodash'; -import { NodeExecutionPhase } from 'models/Execution/enums'; +import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums'; import { TaskNodeMetadata } from 'models/Execution/types'; import * as React from 'react'; import { useNodeExecutionContext } from '../contextProvider/NodeExecutionDetails'; @@ -107,10 +106,7 @@ const DisplayType: React.FC = ({ execution }) => return {type}; }; -const hiddenCacheStatuses = [ - Core.CatalogCacheStatus.CACHE_MISS, - Core.CatalogCacheStatus.CACHE_DISABLED, -]; +const hiddenCacheStatuses = [CatalogCacheStatus.CACHE_MISS, CatalogCacheStatus.CACHE_DISABLED]; function hasCacheStatus(taskNodeMetadata?: TaskNodeMetadata): taskNodeMetadata is TaskNodeMetadata { if (!taskNodeMetadata) { return false; diff --git a/src/components/Executions/Tables/test/NodeExecutionsTable.test.tsx b/src/components/Executions/Tables/test/NodeExecutionsTable.test.tsx index ac3422205..c6e679f37 100644 --- a/src/components/Executions/Tables/test/NodeExecutionsTable.test.tsx +++ b/src/components/Executions/Tables/test/NodeExecutionsTable.test.tsx @@ -20,7 +20,6 @@ import { makeNodeExecutionListQuery } from 'components/Executions/nodeExecutionQ import { NodeExecutionDisplayType } from 'components/Executions/types'; import { nodeExecutionIsTerminal } from 'components/Executions/utils'; import { useConditionalQuery } from 'components/hooks/useConditionalQuery'; -import { Core } from 'flyteidl'; import { cloneDeep } from 'lodash'; import { basicPythonWorkflow } from 'mocks/data/fixtures/basicPythonWorkflow'; import { dynamicExternalSubWorkflow } from 'mocks/data/fixtures/dynamicExternalSubworkflow'; @@ -35,8 +34,9 @@ import { notFoundError } from 'mocks/errors'; import { mockServer } from 'mocks/server'; import { FilterOperationName, RequestConfig } from 'models/AdminEntity/types'; import { nodeExecutionQueryParams } from 'models/Execution/constants'; -import { NodeExecutionPhase } from 'models/Execution/enums'; +import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums'; import { Execution, NodeExecution, TaskNodeMetadata } from 'models/Execution/types'; +import { ResourceType } from 'models/Common/types'; import * as React from 'react'; import { QueryClient, QueryClientProvider, useQueryClient } from 'react-query'; import { makeIdentifier } from 'test/modelUtils'; @@ -225,10 +225,10 @@ describe('NodeExecutionsTable', () => { const { taskExecutions } = nodeExecutions.pythonNode; cachedNodeExecution = nodeExecutions.pythonNode.data; taskNodeMetadata = { - cacheStatus: Core.CatalogCacheStatus.CACHE_MISS, + cacheStatus: CatalogCacheStatus.CACHE_MISS, catalogKey: { datasetId: makeIdentifier({ - resourceType: Core.ResourceType.DATASET, + resourceType: ResourceType.DATASET, }), sourceTaskExecution: { ...taskExecutions.firstAttempt.data.id, @@ -239,12 +239,12 @@ describe('NodeExecutionsTable', () => { }); [ - Core.CatalogCacheStatus.CACHE_HIT, - Core.CatalogCacheStatus.CACHE_LOOKUP_FAILURE, - Core.CatalogCacheStatus.CACHE_POPULATED, - Core.CatalogCacheStatus.CACHE_PUT_FAILURE, + CatalogCacheStatus.CACHE_HIT, + CatalogCacheStatus.CACHE_LOOKUP_FAILURE, + CatalogCacheStatus.CACHE_POPULATED, + CatalogCacheStatus.CACHE_PUT_FAILURE, ].forEach((cacheStatusValue) => - it(`renders correct icon for ${Core.CatalogCacheStatus[cacheStatusValue]}`, async () => { + it(`renders correct icon for ${CatalogCacheStatus[cacheStatusValue]}`, async () => { taskNodeMetadata.cacheStatus = cacheStatusValue; updateNodeExecutions([cachedNodeExecution]); const { getByTitle } = renderTable(); @@ -255,9 +255,9 @@ describe('NodeExecutionsTable', () => { }), ); - [Core.CatalogCacheStatus.CACHE_DISABLED, Core.CatalogCacheStatus.CACHE_MISS].forEach( + [CatalogCacheStatus.CACHE_DISABLED, CatalogCacheStatus.CACHE_MISS].forEach( (cacheStatusValue) => - it(`renders no icon for ${Core.CatalogCacheStatus[cacheStatusValue]}`, async () => { + it(`renders no icon for ${CatalogCacheStatus[cacheStatusValue]}`, async () => { taskNodeMetadata.cacheStatus = cacheStatusValue; updateNodeExecutions([cachedNodeExecution]); const { getByText, queryByTitle } = renderTable(); diff --git a/src/components/Executions/constants.ts b/src/components/Executions/constants.ts index 4af6a7c00..ca70252f1 100644 --- a/src/components/Executions/constants.ts +++ b/src/components/Executions/constants.ts @@ -4,8 +4,8 @@ import { secondaryTextColor, statusColors, } from 'components/Theme/constants'; -import { Core } from 'flyteidl'; import { + CatalogCacheStatus, NodeExecutionPhase, TaskExecutionPhase, WorkflowExecutionPhase, @@ -196,14 +196,13 @@ export const taskTypeToNodeExecutionDisplayType: { [TaskType.ARRAY_K8S]: NodeExecutionDisplayType.ARRAY_K8S, }; -export const cacheStatusMessages: { [k in Core.CatalogCacheStatus]: string } = { - [Core.CatalogCacheStatus.CACHE_DISABLED]: 'Caching was disabled for this execution.', - [Core.CatalogCacheStatus.CACHE_HIT]: 'Output for this execution was read from cache.', - [Core.CatalogCacheStatus.CACHE_LOOKUP_FAILURE]: 'Failed to lookup cache information.', - [Core.CatalogCacheStatus.CACHE_MISS]: 'No cached output was found for this execution.', - [Core.CatalogCacheStatus.CACHE_POPULATED]: 'The result of this execution was written to cache.', - [Core.CatalogCacheStatus.CACHE_PUT_FAILURE]: - 'Failed to write output for this execution to cache.', +export const cacheStatusMessages: { [k in CatalogCacheStatus]: string } = { + [CatalogCacheStatus.CACHE_DISABLED]: 'Caching was disabled for this execution.', + [CatalogCacheStatus.CACHE_HIT]: 'Output for this execution was read from cache.', + [CatalogCacheStatus.CACHE_LOOKUP_FAILURE]: 'Failed to lookup cache information.', + [CatalogCacheStatus.CACHE_MISS]: 'No cached output was found for this execution.', + [CatalogCacheStatus.CACHE_POPULATED]: 'The result of this execution was written to cache.', + [CatalogCacheStatus.CACHE_PUT_FAILURE]: 'Failed to write output for this execution to cache.', }; export const unknownCacheStatusString = 'Cache status is unknown'; export const viewSourceExecutionString = 'View source execution'; diff --git a/src/components/Executions/nodeExecutionQueries.ts b/src/components/Executions/nodeExecutionQueries.ts index c01981475..7d5e3b18b 100644 --- a/src/components/Executions/nodeExecutionQueries.ts +++ b/src/components/Executions/nodeExecutionQueries.ts @@ -243,7 +243,7 @@ async function fetchGroupsForParentNodeExecution( /** GraphUX uses workflowClosure which uses scopedId. This builds a scopedId via parent * nodeExecution to enable mapping between graph and other components */ let scopedId = parentScopeId; - if (scopedId != undefined) { + if (scopedId !== undefined) { scopedId += `-0-${child.metadata?.specNodeId}`; child['scopedId'] = scopedId; } else { diff --git a/src/models/Execution/enums.ts b/src/models/Execution/enums.ts index ea8935845..3209223ca 100644 --- a/src/models/Execution/enums.ts +++ b/src/models/Execution/enums.ts @@ -17,4 +17,6 @@ export type NodeExecutionPhase = Core.NodeExecution.Phase; export const NodeExecutionPhase = Core.NodeExecution.Phase; export type TaskExecutionPhase = Core.TaskExecution.Phase; export const TaskExecutionPhase = Core.TaskExecution.Phase; +export type CatalogCacheStatus = Core.CatalogCacheStatus; +export const CatalogCacheStatus = Core.CatalogCacheStatus; /* eslint-enable @typescript-eslint/no-redeclare */ From 87846daf98c511b8aefd4860547c834042a705fe Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Mon, 28 Mar 2022 14:26:02 -0700 Subject: [PATCH 2/6] chore: update chart bar item tooltip text and position Signed-off-by: Nastya Rusina --- .../Timeline/BarChart/BarChart.stories.tsx | 46 ++++++ .../Timeline/BarChart/barOptions.ts | 60 +++++++ .../BarChart/{chartData.tsx => chartData.ts} | 0 .../Timeline/BarChart/{utils.tsx => utils.ts} | 50 +++++- .../Timeline/ExecutionTimeline.tsx | 13 +- .../Timeline/Timeline.stories.tsx | 156 ------------------ .../ExecutionDetails/Timeline/helpers.ts | 42 ----- .../Timeline/scaleContext.tsx | 13 +- 8 files changed, 155 insertions(+), 225 deletions(-) create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts rename src/components/Executions/ExecutionDetails/Timeline/BarChart/{chartData.tsx => chartData.ts} (100%) rename src/components/Executions/ExecutionDetails/Timeline/BarChart/{utils.tsx => utils.ts} (68%) delete mode 100644 src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx new file mode 100644 index 000000000..2eccbd71f --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx @@ -0,0 +1,46 @@ +import { ComponentMeta, ComponentStory } from '@storybook/react'; +import * as React from 'react'; +import { Bar } from 'react-chartjs-2'; +import { NodeExecutionPhase } from 'models/Execution/enums'; +import { generateChartData, getChartData } from './utils'; +import { getBarOptions } from './barOptions'; + +const isFromCache = [false, true, true, false, false, false]; +const phaseStatus = [ + NodeExecutionPhase.FAILED, + NodeExecutionPhase.SUCCEEDED, + NodeExecutionPhase.SUCCEEDED, + NodeExecutionPhase.RUNNING, + NodeExecutionPhase.UNDEFINED, + NodeExecutionPhase.SUCCEEDED, +]; + +const barItems = [ + { phase: phaseStatus[0], startOffsetSec: 0, durationSec: 5, isFromCache: isFromCache[0] }, + { phase: phaseStatus[1], startOffsetSec: 10, durationSec: 2, isFromCache: isFromCache[1] }, + { phase: phaseStatus[2], startOffsetSec: 0, durationSec: 1, isFromCache: isFromCache[2] }, + { phase: phaseStatus[3], startOffsetSec: 0, durationSec: 10, isFromCache: isFromCache[3] }, + { phase: phaseStatus[4], startOffsetSec: 15, durationSec: 25, isFromCache: isFromCache[4] }, + { phase: phaseStatus[5], startOffsetSec: 7, durationSec: 20, isFromCache: isFromCache[5] }, +]; + +export default { + title: 'Workflow/Timeline', + component: Bar, +} as ComponentMeta; + +const Template: ComponentStory = (args) => ; +export const BarSingle = Template.bind({}); +const phaseDataSingle = generateChartData([barItems[0]]); +BarSingle.args = { + options: getBarOptions(1, phaseDataSingle.tooltipLabel) as any, + data: getChartData(phaseDataSingle), +}; + +export const BarSection = () => { + const phaseData = generateChartData(barItems); + // Chart interval - 1s + return ( + + ); +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts new file mode 100644 index 000000000..d773e0431 --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts @@ -0,0 +1,60 @@ +import { Chart as ChartJS, registerables, Tooltip } from 'chart.js'; +import ChartDataLabels from 'chartjs-plugin-datalabels'; + +ChartJS.register(...registerables, ChartDataLabels); + +// Create positioner to put tooltip at cursor position +Tooltip.positioners.cursor = function (_chartElements, coordinates) { + return coordinates; +}; + +export const getBarOptions = (chartTimeInterval: number, tooltipLabels: string[][]) => { + return { + animation: false as const, + indexAxis: 'y' as const, + elements: { + bar: { + borderWidth: 2, + }, + }, + responsive: true, + maintainAspectRatio: false, + plugins: { + legend: { + display: false, + }, + title: { + display: false, + }, + tooltip: { + // Setting up tooltip: https://www.chartjs.org/docs/latest/configuration/tooltip.html + position: 'cursor', + filter: function (tooltipItem) { + // no tooltip for offsets + return tooltipItem.datasetIndex === 1; + }, + callbacks: { + label: function (context) { + const index = context.dataIndex; + return tooltipLabels[index] ?? ''; + }, + }, + }, + }, + scales: { + x: { + format: Intl.DateTimeFormat, + position: 'top' as const, + ticks: { + display: false, + autoSkip: false, + stepSize: chartTimeInterval, + }, + stacked: true, + }, + y: { + stacked: true, + }, + }, + }; +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts similarity index 100% rename from src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.tsx rename to src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts similarity index 68% rename from src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx rename to src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts index 3a48c637d..627de8a93 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts @@ -5,6 +5,12 @@ import { NodeExecutionPhase } from 'models/Execution/enums'; const CASHED_GREEN = 'rgba(74,227,174,0.25)'; // statusColors.SUCCESS (Mint20) with 25% opacity const TRANSPARENT = 'rgba(0, 0, 0, 0)'; +export enum RelationToCache { + None = 'none', + ReadFromCaceh = 'Read from Cache', + WroteToCache = 'Wrote to cache', +} + export interface BarItemData { phase: NodeExecutionPhase; startOffsetSec: number; @@ -18,10 +24,28 @@ interface ChartDataInput { durations: number[]; startOffset: number[]; offsetColor: string[]; + tooltipLabel: string[][]; barLabel: string[]; barColor: string[]; } +/** + * Depending on amounf of second provided shows data in + * XhXmXs or XmXs or Xs format + */ +export const formatSecondsToHmsFormat = (seconds: number) => { + const hours = Math.floor(seconds / 3600); + seconds %= 3600; + const minutes = Math.floor(seconds / 60); + seconds = seconds % 60; + if (hours > 0) { + return `${hours}h ${minutes}m ${seconds}s`; + } else if (minutes > 0) { + return `${minutes}m ${seconds}s`; + } + return `${seconds}s`; +}; + // narusina - check if exports are still needed export const getOffsetColor = (isCachedValue: boolean[]) => { const colors = isCachedValue.map((val) => (val === true ? CASHED_GREEN : TRANSPARENT)); @@ -35,28 +59,38 @@ export const generateChartData = (data: BarItemData[]): ChartDataInput => { const durations: number[] = []; const startOffset: number[] = []; const offsetColor: string[] = []; + const tooltipLabel: string[][] = []; const barLabel: string[] = []; const barColor: string[] = []; let totalDurationSec = 0; data.forEach((element) => { + const phaseConstant = getNodeExecutionPhaseConstants( + element.phase ?? NodeExecutionPhase.UNDEFINED, + ); + + const durationString = formatSecondsToHmsFormat(element.durationSec); + const tooltipString = `${phaseConstant.text}: ${durationString}`; + // don't show Label if there is now duration yet. + const labelString = element.durationSec > 0 ? durationString : ''; + durations.push(element.durationSec); startOffset.push(element.startOffsetSec); offsetColor.push(element.isFromCache ? CASHED_GREEN : TRANSPARENT); - barLabel.push(element.isFromCache ? 'From cache' : `${Math.round(element.durationSec)}s`); - barColor.push( - getNodeExecutionPhaseConstants(element.phase ?? NodeExecutionPhase.UNDEFINED).badgeColor, - ); + tooltipLabel.push(element.isFromCache ? [tooltipString, 'Read from cache'] : [tooltipString]); + barLabel.push(element.isFromCache ? '\u229A From cache' : labelString); + barColor.push(phaseConstant.badgeColor); totalDurationSec = Math.max(totalDurationSec, element.startOffsetSec + element.durationSec); }); - const elementsNumber = data.length; + return { - elementsNumber, + elementsNumber: data.length, totalDurationSec, durations, startOffset, offsetColor, + tooltipLabel, barLabel, barColor, }; @@ -101,9 +135,7 @@ export const getChartData = (data: ChartDataInput) => { align: 'end' as const, // related to text anchor: 'start' as const, // related to bar formatter: function (value, context) { - return data.offsetColor[context.dataIndex] === CASHED_GREEN - ? '\u229A From Cache' - : `${Math.round(value)}s`; + return data.barLabel[context.dataIndex] ?? ''; }, }, }, diff --git a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx index a8538a5c1..694f66d8f 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx @@ -1,8 +1,6 @@ import * as React from 'react'; import * as moment from 'moment-timezone'; import { Bar } from 'react-chartjs-2'; -import { Chart as ChartJS, registerables } from 'chart.js'; -import ChartDataLabels from 'chartjs-plugin-datalabels'; import { makeStyles, Typography } from '@material-ui/core'; import { useNodeExecutionContext } from 'components/Executions/contextProvider/NodeExecutionDetails'; @@ -14,13 +12,11 @@ import { NodeExecution } from 'models/Execution/types'; import { dNode } from 'models/Graph/types'; import { generateChartData, getChartData } from './BarChart/utils'; import { getChartDurationData } from './BarChart/chartData'; -import { convertToPlainNodes, getBarOptions, TimeZone } from './helpers'; +import { convertToPlainNodes, TimeZone } from './helpers'; import { ChartHeader } from './chartHeader'; import { useScaleContext } from './scaleContext'; import { TaskNames } from './taskNames'; - -// Register components to be usable by chart.js -ChartJS.register(...registerables, ChartDataLabels); +import { getBarOptions } from './BarChart/barOptions'; interface StyleProps { chartWidth: number; @@ -214,7 +210,10 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime
- +
diff --git a/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx b/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx deleted file mode 100644 index 173697dbe..000000000 --- a/src/components/Executions/ExecutionDetails/Timeline/Timeline.stories.tsx +++ /dev/null @@ -1,156 +0,0 @@ -import { ComponentMeta, ComponentStory } from '@storybook/react'; -import * as React from 'react'; -import { Bar } from 'react-chartjs-2'; -import { Chart as ChartJS, registerables } from 'chart.js'; -import ChartDataLabels from 'chartjs-plugin-datalabels'; -import { statusColors } from 'components/Theme/constants'; -import { NodeExecutionPhase } from 'models/Execution/enums'; -import { getNodeExecutionPhaseConstants } from 'components/Executions/utils'; -import { generateChartData, getChartData, getOffsetColor } from './BarChart/utils'; - -ChartJS.register(...registerables, ChartDataLabels); - -const mockData = [5, 2, 10, 25, 20]; -const startOffset = [0, 10, 0, 15, 7]; -const isCachedValue = [false, true, false, false, false]; -const backgroundColor = [ - statusColors.FAILURE, - statusColors.QUEUED, - statusColors.RUNNING, - statusColors.WARNING, - statusColors.SUCCESS, -]; - -const isFromCache = [false, true, true, false, false, false]; -const phaseStatus = [ - NodeExecutionPhase.FAILED, - NodeExecutionPhase.SUCCEEDED, - NodeExecutionPhase.SUCCEEDED, - NodeExecutionPhase.RUNNING, - NodeExecutionPhase.UNDEFINED, - NodeExecutionPhase.SUCCEEDED, -]; - -const barItems = [ - { phase: phaseStatus[0], startOffsetSec: 0, durationSec: 5, isFromCache: isFromCache[0] }, - { phase: phaseStatus[1], startOffsetSec: 10, durationSec: 2, isFromCache: isFromCache[1] }, - { phase: phaseStatus[2], startOffsetSec: 0, durationSec: 1, isFromCache: isFromCache[2] }, - { phase: phaseStatus[3], startOffsetSec: 0, durationSec: 10, isFromCache: isFromCache[3] }, - { phase: phaseStatus[4], startOffsetSec: 15, durationSec: 25, isFromCache: isFromCache[4] }, - { phase: phaseStatus[5], startOffsetSec: 7, durationSec: 20, isFromCache: isFromCache[5] }, -]; - -const chartData = { - labels: mockData.map(() => ''), - datasets: [ - { - data: startOffset, - backgroundColor: getOffsetColor(isCachedValue), - barPercentage: 1, - borderWidth: 0, - datalabels: { - labels: { - title: null, - }, - }, - }, - { - data: mockData.map((duration) => { - return duration === -1 ? 10 : duration === 0 ? 0.5 : duration; - }), - backgroundColor: backgroundColor, - barPercentage: 1, - borderWidth: 0, - datalabels: { - color: '#292936' as const, - align: 'end' as const, - anchor: 'start' as const, - formatter: function (value, context) { - if (mockData[context.dataIndex] === -1) { - return ''; - } - return Math.round(value) + 's'; - }, - }, - }, - ], -}; - -const getBarOptions = ( - chartTimeInterval: number, - isFromCache: boolean[], - phaseStatus: NodeExecutionPhase[], -) => { - return { - animation: false as const, - indexAxis: 'y' as const, - elements: { - bar: { - borderWidth: 2, - }, - }, - responsive: true, - maintainAspectRatio: false, - plugins: { - legend: { - display: false, - }, - title: { - display: false, - }, - tooltip: { - filter: function (tooltipItem) { - // no tooltip for offsets - return tooltipItem.datasetIndex === 1; - }, - // Setting up tooltip: https://www.chartjs.org/docs/latest/configuration/tooltip.html - callbacks: { - label: function (context) { - const index = context.dataIndex; - let label = getNodeExecutionPhaseConstants(phaseStatus[index]).text; - if (context.parsed.x !== null) { - label += `: ${context.parsed.x}s`; - } - - if (isFromCache[index]) { - return [label, 'Read from cache']; - } - return label; - }, - }, - }, - }, - scales: { - x: { - format: Intl.DateTimeFormat, - position: 'top' as const, - ticks: { - display: false, - autoSkip: false, - stepSize: chartTimeInterval, - }, - stacked: true, - }, - y: { - stacked: true, - }, - }, - }; -}; - -export default { - title: 'Workflow/Timeline', - component: Bar, -} as ComponentMeta; - -const Template: ComponentStory = (args) => ; -export const SingleBar = Template.bind({}); -SingleBar.args = { - options: getBarOptions(1, isFromCache, phaseStatus), - data: getChartData(generateChartData(barItems)), -}; - -export const BarSection = () => { - // Chart interval - 1s - return ; -}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/helpers.ts b/src/components/Executions/ExecutionDetails/Timeline/helpers.ts index fad3a5569..ddab892a0 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/helpers.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/helpers.ts @@ -28,45 +28,3 @@ export function convertToPlainNodes(nodes: dNode[], level = 0): dNode[] { }); return result; } - -export const getBarOptions = (chartTimeInterval: number) => { - return { - animation: false as const, - indexAxis: 'y' as const, - elements: { - bar: { - borderWidth: 2, - }, - }, - responsive: true, - maintainAspectRatio: false, - plugins: { - legend: { - display: false, - }, - title: { - display: false, - }, - tooltip: { - filter: function (tooltipItem) { - return tooltipItem.datasetIndex === 1; - }, - }, - }, - scales: { - x: { - format: Intl.DateTimeFormat, - position: 'top' as const, - ticks: { - display: false, - autoSkip: false, - stepSize: chartTimeInterval, - }, - stacked: true, - }, - y: { - stacked: true, - }, - }, - }; -}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/scaleContext.tsx b/src/components/Executions/ExecutionDetails/Timeline/scaleContext.tsx index 6977d23e4..343c82250 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/scaleContext.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/scaleContext.tsx @@ -1,6 +1,7 @@ import { Mark } from '@material-ui/core/Slider'; import * as React from 'react'; import { createContext, useContext } from 'react'; +import { formatSecondsToHmsFormat } from './BarChart/utils'; const MIN_SCALE_VALUE = 60; // 1 min const MAX_SCALE_VALUE = 3600; // 1h @@ -36,16 +37,6 @@ export const ScaleContext = createContext({ /** Could be used to access the whole TimelineScaleState */ export const useScaleContext = (): TimelineScaleState => useContext(ScaleContext); -const formatSeconds = (time: number) => { - if (time < 60) { - return `${time}s`; - } - if (time % 60 === 0) { - return `${Math.floor(time / 60)}m`; - } - return `${Math.floor(time / 60)}m ${time % 60}s`; -}; - interface ScaleProviderProps { children?: React.ReactNode; } @@ -69,7 +60,7 @@ export const ScaleProvider = (props: ScaleProviderProps) => { for (let i = 0; i < percentage.length; ++i) { newMarks.push({ value: i, - label: formatSeconds(getIntervalValue(i)), + label: formatSecondsToHmsFormat(getIntervalValue(i)), }); } setMarks(newMarks); From 9ecf4d04d6798bd1075e7bf5d1b6a1d609f88c63 Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Mon, 28 Mar 2022 16:22:27 -0700 Subject: [PATCH 3/6] chore: add loading state + simplify storybook Signed-off-by: Nastya Rusina --- .../Timeline/BarChart/BarChart.stories.tsx | 49 +++++------------- .../BarChart/BarChartSingleItem.stories.tsx | 49 ++++++++++++++++++ .../Timeline/BarChart/barOptions.ts | 4 +- .../Timeline/BarChart/chartData.ts | 14 +++-- .../Timeline/BarChart/index.tsx | 20 ++++++++ .../Timeline/BarChart/utils.ts | 5 -- .../Timeline/ExecutionTimeline.tsx | 51 +++++++------------ .../ExecutionDetails/Timeline/chartHeader.tsx | 29 +++++++++-- .../ExecutionDetails/Timeline/index.tsx | 18 ++++++- 9 files changed, 155 insertions(+), 84 deletions(-) create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChartSingleItem.stories.tsx create mode 100644 src/components/Executions/ExecutionDetails/Timeline/BarChart/index.tsx diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx index 2eccbd71f..53574c927 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChart.stories.tsx @@ -1,46 +1,25 @@ import { ComponentMeta, ComponentStory } from '@storybook/react'; import * as React from 'react'; -import { Bar } from 'react-chartjs-2'; import { NodeExecutionPhase } from 'models/Execution/enums'; -import { generateChartData, getChartData } from './utils'; -import { getBarOptions } from './barOptions'; - -const isFromCache = [false, true, true, false, false, false]; -const phaseStatus = [ - NodeExecutionPhase.FAILED, - NodeExecutionPhase.SUCCEEDED, - NodeExecutionPhase.SUCCEEDED, - NodeExecutionPhase.RUNNING, - NodeExecutionPhase.UNDEFINED, - NodeExecutionPhase.SUCCEEDED, -]; +import { BarChart } from '.'; const barItems = [ - { phase: phaseStatus[0], startOffsetSec: 0, durationSec: 5, isFromCache: isFromCache[0] }, - { phase: phaseStatus[1], startOffsetSec: 10, durationSec: 2, isFromCache: isFromCache[1] }, - { phase: phaseStatus[2], startOffsetSec: 0, durationSec: 1, isFromCache: isFromCache[2] }, - { phase: phaseStatus[3], startOffsetSec: 0, durationSec: 10, isFromCache: isFromCache[3] }, - { phase: phaseStatus[4], startOffsetSec: 15, durationSec: 25, isFromCache: isFromCache[4] }, - { phase: phaseStatus[5], startOffsetSec: 7, durationSec: 20, isFromCache: isFromCache[5] }, + { phase: NodeExecutionPhase.FAILED, startOffsetSec: 0, durationSec: 5, isFromCache: false }, + { phase: NodeExecutionPhase.SUCCEEDED, startOffsetSec: 10, durationSec: 2, isFromCache: true }, + { phase: NodeExecutionPhase.SUCCEEDED, startOffsetSec: 0, durationSec: 1, isFromCache: true }, + { phase: NodeExecutionPhase.RUNNING, startOffsetSec: 0, durationSec: 10, isFromCache: false }, + { phase: NodeExecutionPhase.UNDEFINED, startOffsetSec: 15, durationSec: 25, isFromCache: false }, + { phase: NodeExecutionPhase.SUCCEEDED, startOffsetSec: 7, durationSec: 20, isFromCache: false }, ]; export default { title: 'Workflow/Timeline', - component: Bar, -} as ComponentMeta; - -const Template: ComponentStory = (args) => ; -export const BarSingle = Template.bind({}); -const phaseDataSingle = generateChartData([barItems[0]]); -BarSingle.args = { - options: getBarOptions(1, phaseDataSingle.tooltipLabel) as any, - data: getChartData(phaseDataSingle), -}; + component: BarChart, +} as ComponentMeta; -export const BarSection = () => { - const phaseData = generateChartData(barItems); - // Chart interval - 1s - return ( - - ); +const Template: ComponentStory = (args) => ; +export const BarSection = Template.bind({}); +BarSection.args = { + items: barItems, + chartTimeIntervalSec: 1, }; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChartSingleItem.stories.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChartSingleItem.stories.tsx new file mode 100644 index 000000000..481e2fb78 --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/BarChartSingleItem.stories.tsx @@ -0,0 +1,49 @@ +import { ComponentMeta, ComponentStory } from '@storybook/react'; +import * as React from 'react'; +import { NodeExecutionPhase } from 'models/Execution/enums'; +import { BarItemData } from './utils'; +import { BarChart } from '.'; + +const phaseEnumTyping = { + options: Object.values(NodeExecutionPhase), + mapping: Object.values(NodeExecutionPhase), + control: { + type: 'select', + labels: Object.keys(NodeExecutionPhase), + }, +}; + +interface SingleItemProps extends BarItemData { + chartTimeIntervalSec: number; +} + +/** + * This is a fake storybook only component, to allow ease experimentation whith single bar item + */ +const SingleBarItem = (props: SingleItemProps) => { + const items = [props]; + return ; +}; + +export default { + title: 'Workflow/Timeline', + component: SingleBarItem, + // 👇 Creates specific argTypes + argTypes: { + phase: phaseEnumTyping, + }, +} as ComponentMeta; + +const TemplateSingleItem: ComponentStory = (args) => ( + +); + +export const BarChartSingleItem = TemplateSingleItem.bind({}); +// const phaseDataSingle = generateChartData([barItems[0]]); +BarChartSingleItem.args = { + phase: NodeExecutionPhase.ABORTED, + startOffsetSec: 15, + durationSec: 30, + isFromCache: false, + chartTimeIntervalSec: 5, +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts index d773e0431..99f50400a 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/barOptions.ts @@ -8,7 +8,7 @@ Tooltip.positioners.cursor = function (_chartElements, coordinates) { return coordinates; }; -export const getBarOptions = (chartTimeInterval: number, tooltipLabels: string[][]) => { +export const getBarOptions = (chartTimeIntervalSec: number, tooltipLabels: string[][]) => { return { animation: false as const, indexAxis: 'y' as const, @@ -48,7 +48,7 @@ export const getBarOptions = (chartTimeInterval: number, tooltipLabels: string[] ticks: { display: false, autoSkip: false, - stepSize: chartTimeInterval, + stepSize: chartTimeIntervalSec, }, stacked: true, }, diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts index 801471f45..3bb12b854 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts @@ -10,9 +10,13 @@ const EMPTY_BAR_ITEM: BarItemData = { isFromCache: false, }; -export const getChartDurationData = (nodes: dNode[], startedAt: Date): BarItemData[] => { - if (nodes.length === 0) return []; +export const getChartDurationData = ( + nodes: dNode[], + startedAt: Date, +): { items: BarItemData[]; totalDurationSec: number } => { + if (nodes.length === 0) return { items: [], totalDurationSec: 0 }; + let totalDurationSec = 0; const initialStartTime = startedAt.getTime(); const result: BarItemData[] = nodes.map(({ execution }) => { if (!execution) { @@ -51,9 +55,11 @@ export const getChartDurationData = (nodes: dNode[], startedAt: Date): BarItemDa durationSec = execution.closure.duration?.seconds?.toNumber() ?? 0; } - return { phase, startOffsetSec: startOffset / 1000, durationSec, isFromCache }; + const startOffsetSec = startOffset / 1000; + totalDurationSec = Math.max(totalDurationSec, startOffsetSec + durationSec); + return { phase, startOffsetSec, durationSec, isFromCache }; }); // Do we want to get initialStartTime from different place, to avoid recalculating it. - return result; + return { items: result, totalDurationSec }; }; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/index.tsx b/src/components/Executions/ExecutionDetails/Timeline/BarChart/index.tsx new file mode 100644 index 000000000..903d96f9a --- /dev/null +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/index.tsx @@ -0,0 +1,20 @@ +import * as React from 'react'; +import { Bar } from 'react-chartjs-2'; +import { getBarOptions } from './barOptions'; +import { BarItemData, generateChartData, getChartData } from './utils'; + +interface BarChartProps { + items: BarItemData[]; + chartTimeIntervalSec: number; +} + +export const BarChart = (props: BarChartProps) => { + const phaseData = generateChartData(props.items); + + return ( + + ); +}; diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts index 627de8a93..bcefb3496 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts @@ -20,7 +20,6 @@ export interface BarItemData { interface ChartDataInput { elementsNumber: number; - totalDurationSec: number; durations: number[]; startOffset: number[]; offsetColor: string[]; @@ -63,7 +62,6 @@ export const generateChartData = (data: BarItemData[]): ChartDataInput => { const barLabel: string[] = []; const barColor: string[] = []; - let totalDurationSec = 0; data.forEach((element) => { const phaseConstant = getNodeExecutionPhaseConstants( element.phase ?? NodeExecutionPhase.UNDEFINED, @@ -80,13 +78,10 @@ export const generateChartData = (data: BarItemData[]): ChartDataInput => { tooltipLabel.push(element.isFromCache ? [tooltipString, 'Read from cache'] : [tooltipString]); barLabel.push(element.isFromCache ? '\u229A From cache' : labelString); barColor.push(phaseConstant.badgeColor); - - totalDurationSec = Math.max(totalDurationSec, element.startOffsetSec + element.durationSec); }); return { elementsNumber: data.length, - totalDurationSec, durations, startOffset, offsetColor, diff --git a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx index 694f66d8f..08dad1406 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/ExecutionTimeline.tsx @@ -1,6 +1,4 @@ import * as React from 'react'; -import * as moment from 'moment-timezone'; -import { Bar } from 'react-chartjs-2'; import { makeStyles, Typography } from '@material-ui/core'; import { useNodeExecutionContext } from 'components/Executions/contextProvider/NodeExecutionDetails'; @@ -10,13 +8,12 @@ import { tableHeaderColor } from 'components/Theme/constants'; import { timestampToDate } from 'common/utils'; import { NodeExecution } from 'models/Execution/types'; import { dNode } from 'models/Graph/types'; -import { generateChartData, getChartData } from './BarChart/utils'; import { getChartDurationData } from './BarChart/chartData'; -import { convertToPlainNodes, TimeZone } from './helpers'; +import { convertToPlainNodes } from './helpers'; import { ChartHeader } from './chartHeader'; import { useScaleContext } from './scaleContext'; import { TaskNames } from './taskNames'; -import { getBarOptions } from './BarChart/barOptions'; +import { BarChart } from './BarChart'; interface StyleProps { chartWidth: number; @@ -86,6 +83,7 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime const [startedAt, setStartedAt] = React.useState(new Date()); const { compiledWorkflowClosure } = useNodeExecutionContext(); + const { chartInterval: chartTimeInterval } = useScaleContext(); React.useEffect(() => { const nodes: dNode[] = compiledWorkflowClosure @@ -114,29 +112,22 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime } }, [originalNodes, nodeExecutions]); - const barItemsData = getChartDurationData(showNodes, startedAt); - const chartDataInput = generateChartData(barItemsData); - const { chartInterval: chartTimeInterval, setMaxValue } = useScaleContext(); - const styles = useStyles({ chartWidth: chartWidth, itemsShown: chartDataInput.elementsNumber }); + const { items: barItemsData, totalDurationSec } = getChartDurationData(showNodes, startedAt); + const styles = useStyles({ chartWidth: chartWidth, itemsShown: showNodes.length }); React.useEffect(() => { - setMaxValue(chartDataInput.totalDurationSec); - }, [chartDataInput.totalDurationSec, setMaxValue]); - - React.useEffect(() => { - const calcWidth = - Math.ceil(chartDataInput.totalDurationSec / chartTimeInterval) * INTERVAL_LENGTH; + // Sync width of elements and intervals of ChartHeader (time labels) and BarChart + const calcWidth = Math.ceil(totalDurationSec / chartTimeInterval) * INTERVAL_LENGTH; if (durationsRef.current && calcWidth < durationsRef.current.clientWidth) { setLabelInterval( - durationsRef.current.clientWidth / - Math.ceil(chartDataInput.totalDurationSec / chartTimeInterval), + durationsRef.current.clientWidth / Math.ceil(totalDurationSec / chartTimeInterval), ); setChartWidth(durationsRef.current.clientWidth); } else { setChartWidth(calcWidth); setLabelInterval(INTERVAL_LENGTH); } - }, [chartDataInput.totalDurationSec, chartTimeInterval, durationsRef]); + }, [totalDurationSec, chartTimeInterval, durationsRef]); const onGraphScroll = () => { // cover horizontal scroll only @@ -182,17 +173,6 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime setOriginalNodes([...originalNodes]); }; - const labels = React.useMemo(() => { - const len = Math.ceil(chartDataInput.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, chartDataInput.totalDurationSec]); - return ( <>
@@ -206,14 +186,17 @@ export const ExecutionTimeline: React.FC = ({ nodeExecutions, chartTime
- +
- +
diff --git a/src/components/Executions/ExecutionDetails/Timeline/chartHeader.tsx b/src/components/Executions/ExecutionDetails/Timeline/chartHeader.tsx index da56b6c63..1fafd08c8 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/chartHeader.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/chartHeader.tsx @@ -1,6 +1,9 @@ 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; @@ -25,17 +28,37 @@ const useStyles = makeStyles((_theme) => ({ })); interface HeaderProps extends StyleProps { - labels: string[]; + 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 (
- {props.labels.map((label, idx) => { + {labels.map((label) => { return ( -
+
{label}
); diff --git a/src/components/Executions/ExecutionDetails/Timeline/index.tsx b/src/components/Executions/ExecutionDetails/Timeline/index.tsx index 70792e2ed..e0349ddab 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/index.tsx +++ b/src/components/Executions/ExecutionDetails/Timeline/index.tsx @@ -6,6 +6,7 @@ import { NodeExecutionsRequestConfigContext } from 'components/Executions/contex import { useAllTreeNodeExecutionGroupsQuery } from 'components/Executions/nodeExecutionQueries'; import { DataError } from 'components/Errors/DataError'; import { DetailsPanel } from 'components/common/DetailsPanel'; +import { LargeLoadingSpinner } from 'components/common/LoadingSpinner'; import { NodeExecutionDetailsPanelContent } from '../NodeExecutionDetailsPanelContent'; import { NodeExecutionsTimelineContext } from './context'; import { ExecutionTimelineFooter } from './ExecutionTimelineFooter'; @@ -24,6 +25,9 @@ const useStyles = makeStyles(() => ({ flex: '1 1 0', overflowY: 'auto', }, + loading: { + margin: 'auto', + }, })); interface TimelineProps { @@ -53,12 +57,24 @@ export const ExecutionNodesTimeline = (props: TimelineProps) => { return ; }; + const TimelineLoading = () => { + return ( +
+ +
+ ); + }; + return (
- + {renderExecutionsTimeline} From bc7cdff4cf086acd78272b36c974214a5a4d5eb1 Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Mon, 28 Mar 2022 17:20:14 -0700 Subject: [PATCH 4/6] chore: add basic test Signed-off-by: Nastya Rusina --- .../ExecutionDetails/test/BarChart.test.tsx | 22 +++++++++++++++++++ .../test/ExecutionNodeViews.test.tsx | 1 + .../ExecutionDetails/test/Timeline.test.tsx | 1 + 3 files changed, 24 insertions(+) create mode 100644 src/components/Executions/ExecutionDetails/test/BarChart.test.tsx diff --git a/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx b/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx new file mode 100644 index 000000000..0adff642d --- /dev/null +++ b/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx @@ -0,0 +1,22 @@ +import { formatSecondsToHmsFormat } from '../Timeline/BarChart/utils'; + +describe('ExecutionDetails > Timeline > BarChart', () => { + it('formatSecondsToHmsFormat works as expected', () => { + // more than hour + expect(formatSecondsToHmsFormat(4231)).toEqual('1h 10m 31s'); + expect(formatSecondsToHmsFormat(3601)).toEqual('1h 0m 1s'); + // 1 hour + expect(formatSecondsToHmsFormat(3600)).toEqual('1h 0m 0s'); + + // less than 1 hour and more than 1 minute + expect(formatSecondsToHmsFormat(3599)).toEqual('59m 59s'); + expect(formatSecondsToHmsFormat(600)).toEqual('10m 0s'); + expect(formatSecondsToHmsFormat(61)).toEqual('1m 1s'); + // 1 minute + expect(formatSecondsToHmsFormat(60)).toEqual('1m 0s'); + // less than minute + expect(formatSecondsToHmsFormat(59)).toEqual('59s'); + expect(formatSecondsToHmsFormat(23)).toEqual('23s'); + expect(formatSecondsToHmsFormat(0)).toEqual('0s'); + }); +}); diff --git a/src/components/Executions/ExecutionDetails/test/ExecutionNodeViews.test.tsx b/src/components/Executions/ExecutionDetails/test/ExecutionNodeViews.test.tsx index 3b11118f8..874dc7557 100644 --- a/src/components/Executions/ExecutionDetails/test/ExecutionNodeViews.test.tsx +++ b/src/components/Executions/ExecutionDetails/test/ExecutionNodeViews.test.tsx @@ -19,6 +19,7 @@ jest.mock('../ExecutionWorkflowGraph.tsx', () => ({ jest.mock('chart.js', () => ({ Chart: { register: () => null }, + Tooltip: { positioners: { cursor: () => null } }, registerables: [], })); diff --git a/src/components/Executions/ExecutionDetails/test/Timeline.test.tsx b/src/components/Executions/ExecutionDetails/test/Timeline.test.tsx index 31949aa4d..d1e68dba2 100644 --- a/src/components/Executions/ExecutionDetails/test/Timeline.test.tsx +++ b/src/components/Executions/ExecutionDetails/test/Timeline.test.tsx @@ -18,6 +18,7 @@ jest.mock('../ExecutionWorkflowGraph.tsx', () => ({ jest.mock('chart.js', () => ({ Chart: { register: () => null }, + Tooltip: { positioners: { cursor: () => null } }, registerables: [], })); From 2541edfb4344f2ddfb0094f896a60f97e72c9af8 Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Tue, 29 Mar 2022 10:57:24 -0700 Subject: [PATCH 5/6] chore: more tests Signed-off-by: Nastya Rusina --- .../Timeline/BarChart/utils.ts | 4 +- .../ExecutionDetails/test/BarChart.test.tsx | 48 +++++++++++- .../test/__mocks__/NodeExecution.mock.ts | 77 +++++++++++++++++++ 3 files changed, 126 insertions(+), 3 deletions(-) create mode 100644 src/components/Executions/ExecutionDetails/test/__mocks__/NodeExecution.mock.ts diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts index bcefb3496..a8b1d9510 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/utils.ts @@ -2,8 +2,8 @@ import { getNodeExecutionPhaseConstants } from 'components/Executions/utils'; import { primaryTextColor } from 'components/Theme/constants'; import { NodeExecutionPhase } from 'models/Execution/enums'; -const CASHED_GREEN = 'rgba(74,227,174,0.25)'; // statusColors.SUCCESS (Mint20) with 25% opacity -const TRANSPARENT = 'rgba(0, 0, 0, 0)'; +export const CASHED_GREEN = 'rgba(74,227,174,0.25)'; // statusColors.SUCCESS (Mint20) with 25% opacity +export const TRANSPARENT = 'rgba(0, 0, 0, 0)'; export enum RelationToCache { None = 'none', diff --git a/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx b/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx index 0adff642d..5d8270b4b 100644 --- a/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx +++ b/src/components/Executions/ExecutionDetails/test/BarChart.test.tsx @@ -1,4 +1,12 @@ -import { formatSecondsToHmsFormat } from '../Timeline/BarChart/utils'; +import { getChartDurationData } from '../Timeline/BarChart/chartData'; +import { + CASHED_GREEN, + formatSecondsToHmsFormat, + generateChartData, + getOffsetColor, + TRANSPARENT, +} from '../Timeline/BarChart/utils'; +import { getMockExecutionsForBarChart, mockbarItems } from './__mocks__/NodeExecution.mock'; describe('ExecutionDetails > Timeline > BarChart', () => { it('formatSecondsToHmsFormat works as expected', () => { @@ -19,4 +27,42 @@ describe('ExecutionDetails > Timeline > BarChart', () => { expect(formatSecondsToHmsFormat(23)).toEqual('23s'); expect(formatSecondsToHmsFormat(0)).toEqual('0s'); }); + + it('getOffsetColor returns colored background for cached items', () => { + const cachedArray = [false, true, false]; + const offsetColors = getOffsetColor(cachedArray); + + // If items is not cached - offset is transparent + expect(offsetColors[0]).toEqual(TRANSPARENT); + expect(offsetColors[2]).toEqual(TRANSPARENT); + // If cached - colored backfground + expect(offsetColors[1]).toEqual(CASHED_GREEN); + }); + + // Mock bars used below + // const mockbarItems = [ + // { phase: NodeExecutionPhase.FAILED, startOffsetSec: 0, durationSec: 15, isFromCache: false }, + // { phase: NodeExecutionPhase.SUCCEEDED, startOffsetSec: 5, durationSec: 11, isFromCache: true }, + // { phase: NodeExecutionPhase.RUNNING, startOffsetSec: 17, durationSec: 23, isFromCache: false }, + // { phase: NodeExecutionPhase.QUEUED, startOffsetSec: 39, durationSec: 0, isFromCache: false }, + // ]; + // it('getChartDurationData is properly generated from Node[] items', () => { + // /** */ + // const startTime = 1642627611; + // const mockData = getMockExecutionsForBarChart(startTime); + // const chartItems = getChartDurationData(mockData, new Date(1642627611 * 1000)); + + // expect(chartItems[0]).toEqual(mockbarItems[0]); + // }); + + it('generateChartData properly generates map of data for ChartBars', () => { + const chartData = generateChartData(mockbarItems); + expect(chartData.durations).toEqual([15, 11, 23, 0]); + expect(chartData.startOffset).toEqual([0, 5, 17, 39]); + expect(chartData.offsetColor).toEqual([TRANSPARENT, CASHED_GREEN, TRANSPARENT, TRANSPARENT]); + // labels looks as expected + expect(chartData.barLabel[0]).toEqual(formatSecondsToHmsFormat(mockbarItems[0].durationSec)); + expect(chartData.barLabel[1]).toEqual('\u229A From cache'); + expect(chartData.barLabel[3]).toEqual(''); + }); }); diff --git a/src/components/Executions/ExecutionDetails/test/__mocks__/NodeExecution.mock.ts b/src/components/Executions/ExecutionDetails/test/__mocks__/NodeExecution.mock.ts new file mode 100644 index 000000000..d5a0a33f5 --- /dev/null +++ b/src/components/Executions/ExecutionDetails/test/__mocks__/NodeExecution.mock.ts @@ -0,0 +1,77 @@ +import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums'; + +const dNodeBasicExecution = { + id: 'other-root-n0', + scopedId: 'other-root-n0', + execution: { + id: { + nodeId: 'other-root-n0', + executionId: { project: 'flytesnacks', domain: 'development', name: 'rnktdb3skr' }, + }, + closure: { + phase: 3, + startedAt: { seconds: { low: 1642627611, high: 0, unsigned: false }, nanos: 0 }, + duration: { seconds: { low: 55, high: 0, unsigned: false }, nanos: 0 }, + createdAt: { seconds: { low: 1642627611, high: 0, unsigned: false }, nanos: 0 }, + updatedAt: { seconds: { low: 1642627666, high: 0, unsigned: false }, nanos: 0 }, + outputUri: + 's3://flyte-demo/metadata/propeller/flytesnacks-development-rnktdb3skr/other-root-n0/data/0/outputs.pb', + }, + metadata: { isParentNode: true, specNodeId: 'other-root-n0' }, + scopedId: 'other-root-n0', + }, +}; + +const getMockNodeExecution = ( + initialStartSec: number, + phase: NodeExecutionPhase, + startOffsetSec: number, + durationSec: number, + cacheStatus?: CatalogCacheStatus, +) => { + const node = { ...dNodeBasicExecution } as any; + node.execution.closure.phase = phase; + if (cacheStatus) { + node.execution.closure = { + ...node.execution.closure, + taskNodeMetadata: { + cacheStatus: cacheStatus, + }, + }; + if (cacheStatus === CatalogCacheStatus.CACHE_HIT) { + node.execution.closure.createdAt.seconds.low = initialStartSec + startOffsetSec; + node.execution.closure.updatedAt.seconds.low = initialStartSec + startOffsetSec + durationSec; + return { + ...node, + execution: { + ...node.execution, + closure: { + ...node.execution.closure, + startedAt: undefined, + duration: undefined, + }, + }, + }; + } + } + node.execution.closure.startedAt.seconds.low = initialStartSec + startOffsetSec; + node.execution.closure.duration.seconds.low = initialStartSec + startOffsetSec + durationSec; + return node; +}; + +export const mockbarItems = [ + { phase: NodeExecutionPhase.FAILED, startOffsetSec: 0, durationSec: 15, isFromCache: false }, + { phase: NodeExecutionPhase.SUCCEEDED, startOffsetSec: 5, durationSec: 11, isFromCache: true }, + { phase: NodeExecutionPhase.RUNNING, startOffsetSec: 17, durationSec: 23, isFromCache: false }, + { phase: NodeExecutionPhase.QUEUED, startOffsetSec: 39, durationSec: 0, isFromCache: false }, +]; + +export const getMockExecutionsForBarChart = (startTimeSec: number) => { + const start = startTimeSec; + return [ + getMockNodeExecution(start, NodeExecutionPhase.FAILED, 0, 15), + getMockNodeExecution(start, NodeExecutionPhase.SUCCEEDED, 5, 11, CatalogCacheStatus.CACHE_HIT), + getMockNodeExecution(start, NodeExecutionPhase.RUNNING, 17, 23, CatalogCacheStatus.CACHE_MISS), + getMockNodeExecution(start, NodeExecutionPhase.QUEUED, 39, 0), + ]; +}; From a26b6cd661f8d3d04f962bce6d099e7e7f461ced Mon Sep 17 00:00:00 2001 From: Nastya Rusina Date: Tue, 29 Mar 2022 12:24:45 -0700 Subject: [PATCH 6/6] fix: partial fix to avoid aborted tasks carsh wrongly marked as running Signed-off-by: Nastya Rusina --- .../Timeline/BarChart/chartData.ts | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts index 3bb12b854..adbf5e8bf 100644 --- a/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts +++ b/src/components/Executions/ExecutionDetails/Timeline/BarChart/chartData.ts @@ -3,6 +3,8 @@ import { CatalogCacheStatus, NodeExecutionPhase } from 'models/Execution/enums'; import { dNode } from 'models/Graph/types'; import { BarItemData } from './utils'; +const WEEK_DURATION_SEC = 7 * 24 * 3600; + const EMPTY_BAR_ITEM: BarItemData = { phase: NodeExecutionPhase.UNDEFINED, startOffsetSec: 0, @@ -23,7 +25,7 @@ export const getChartDurationData = ( return EMPTY_BAR_ITEM; } - const phase = execution.closure.phase; + let phase = execution.closure.phase; const isFromCache = execution.closure.taskNodeMetadata?.cacheStatus === CatalogCacheStatus.CACHE_HIT; @@ -46,16 +48,24 @@ export const getChartDurationData = ( durationSec = updatedAt - createdAt; durationSec = durationSec === 0 ? 2 : durationSec; } else if (phase === NodeExecutionPhase.RUNNING) { - // we need to add check if parents are failed - workflow status - ? if (startedAt) { const duration = Date.now() - timestampToDate(startedAt).getTime(); durationSec = duration / 1000; + if (durationSec > WEEK_DURATION_SEC) { + // TODO: https://github.com/flyteorg/flyteconsole/issues/332 + // In some cases tasks which were needed to be ABORTED are stuck in running state, + // In case if task is still running after a week - we assume it should have been aborted. + // The proper fix should be covered by isue: flyteconsole#332 + phase = NodeExecutionPhase.ABORTED; + const allegedDurationSec = Math.trunc(totalDurationSec - startOffset / 1000); + durationSec = allegedDurationSec > 0 ? allegedDurationSec : 10; + } } } else { durationSec = execution.closure.duration?.seconds?.toNumber() ?? 0; } - const startOffsetSec = startOffset / 1000; + const startOffsetSec = Math.trunc(startOffset / 1000); totalDurationSec = Math.max(totalDurationSec, startOffsetSec + durationSec); return { phase, startOffsetSec, durationSec, isFromCache }; });