From f0a3183a4af032ba37845fcd468a9afec7867447 Mon Sep 17 00:00:00 2001 From: "Ricardo M." Date: Mon, 9 Dec 2024 09:19:18 +0100 Subject: [PATCH] feat(Canvas): Use `path`+`scope` as VizNode ID Currently, when transforming a flow into `VisualizationNodes`, the nodes `Id`s are randomly generated, causing that whenever there's a change in the structure of the flow, previous `Id`s change. This commit uses the `path` + `scope` so the `VisualizationNodes` IDs are stable between interacting with the flow. --- .../hoverToolbarActions.cy.ts | 10 +- .../branchingStepAddition.cy.ts | 12 +- .../mandatoryPropsWarnings.cy.ts | 10 +- .../cypress/support/next-commands/design.ts | 4 +- .../Visualization/Canvas/Canvas.tsx | 2 +- .../Canvas/CanvasSideBar.test.tsx | 2 +- .../Canvas/Form/CanvasForm.test.tsx | 14 +- .../__snapshots__/CanvasForm.test.tsx.snap | 2 +- .../Canvas/__snapshots__/Canvas.test.tsx.snap | 736 +++++++++++++----- .../__snapshots__/CanvasSideBar.test.tsx.snap | 2 +- .../__snapshots__/flow.service.test.ts.snap | 75 +- .../Visualization/Canvas/flow.service.test.ts | 16 +- .../Visualization/Canvas/flow.service.ts | 14 +- .../ContextMenu/ItemDeleteGroup.test.tsx | 2 +- .../ContextMenu/ItemEnableAllSteps.test.tsx | 4 +- .../ContextMenu/NodeContextMenu.test.tsx | 2 +- .../flows/nodes/mappers/base-node-mapper.ts | 7 +- .../flows/nodes/mappers/choice-node-mapper.ts | 2 +- .../nodes/mappers/otherwise-node-mapper.ts | 2 +- .../parallel-processor-base-node-mapper.ts | 2 +- .../flows/nodes/mappers/when-node-mapper.ts | 2 +- .../visualization/visualization-node.test.ts | 20 +- .../visualization/visualization-node.ts | 3 +- .../__snapshots__/nodes-edges.test.ts.snap | 344 ++++---- packages/ui/src/tests/nodes-edges.test.ts | 2 +- .../src/utils/get-viznodes-from-graph.test.ts | 6 +- 26 files changed, 835 insertions(+), 462 deletions(-) diff --git a/packages/ui-tests/cypress/e2e/designer/basicNodeActions/hoverToolbarActions.cy.ts b/packages/ui-tests/cypress/e2e/designer/basicNodeActions/hoverToolbarActions.cy.ts index 61f8adbfd..8a2e1792f 100644 --- a/packages/ui-tests/cypress/e2e/designer/basicNodeActions/hoverToolbarActions.cy.ts +++ b/packages/ui-tests/cypress/e2e/designer/basicNodeActions/hoverToolbarActions.cy.ts @@ -34,14 +34,20 @@ describe('Test toolbar on hover actions', () => { cy.get('[data-testid="step-toolbar-button-disable"]').click(); cy.openStepConfigurationTab('setHeader'); + + // Temporary workaround since the toolbar is updated but the config form is closed + cy.openStepConfigurationTab('setHeader'); + cy.selectFormTab('All'); cy.checkConfigCheckboxObject('disabled', true); - cy.openStepConfigurationTab('setHeader'); cy.get('[data-testid="step-toolbar-button-disable"]').click(); cy.openStepConfigurationTab('setHeader'); - cy.selectFormTab('All'); + + // Temporary workaround since the toolbar is updated but the config form is closed + cy.openStepConfigurationTab('setHeader'); + cy.checkConfigCheckboxObject('disabled', false); }); diff --git a/packages/ui-tests/cypress/e2e/designer/branchingFlows/branchingStepAddition.cy.ts b/packages/ui-tests/cypress/e2e/designer/branchingFlows/branchingStepAddition.cy.ts index 9b4f815ea..a2c2b80c4 100644 --- a/packages/ui-tests/cypress/e2e/designer/branchingFlows/branchingStepAddition.cy.ts +++ b/packages/ui-tests/cypress/e2e/designer/branchingFlows/branchingStepAddition.cy.ts @@ -61,7 +61,10 @@ describe('Test for Branching actions from the canvas', () => { cy.chooseFromCatalog('component', 'activemq'); cy.checkNodeExist('activemq', 1); - cy.checkEdgeExists('setHeader', 'activemq'); + cy.checkEdgeExists( + 'template.from.steps.1.choice.when.0.steps.1.setHeader', + 'template.from.steps.1.choice.when.0.steps.2.to', + ); }); it('User prepends a step in a branch from the canvas (first in the branch)', () => { @@ -73,7 +76,10 @@ describe('Test for Branching actions from the canvas', () => { cy.chooseFromCatalog('component', 'activemq'); cy.checkNodeExist('activemq', 1); - cy.checkEdgeExists('activemq', 'digitalocean'); + cy.checkEdgeExists( + 'template.from.steps.1.choice.when.0.steps.0.to', + 'template.from.steps.1.choice.when.0.steps.1.to', + ); }); it('User prepends a step to a step whose previous step contains branches', () => { @@ -85,6 +91,6 @@ describe('Test for Branching actions from the canvas', () => { cy.chooseFromCatalog('component', 'activemq'); cy.checkNodeExist('activemq', 1); - cy.checkEdgeExists('activemq', 'filter'); + cy.checkEdgeExists('template.from.steps.2.to', 'template.from.steps.3.filter'); }); }); diff --git a/packages/ui-tests/cypress/e2e/designer/propsWarnings/mandatoryPropsWarnings.cy.ts b/packages/ui-tests/cypress/e2e/designer/propsWarnings/mandatoryPropsWarnings.cy.ts index 4cf9548f1..72289b7ba 100644 --- a/packages/ui-tests/cypress/e2e/designer/propsWarnings/mandatoryPropsWarnings.cy.ts +++ b/packages/ui-tests/cypress/e2e/designer/propsWarnings/mandatoryPropsWarnings.cy.ts @@ -12,7 +12,7 @@ describe('Test for missing config props canvas warnings', () => { cy.checkNodeExist('github', 1); - cy.get('[data-id^="github"] g') + cy.get('[data-id^="camel-route|route.from.steps.1.to"] g') .find('span[data-warning="true"].pf-v5-c-icon') .should('have.attr', 'title', '3 required parameters are not yet configured: [ type,repoName,repoOwner ]'); @@ -21,7 +21,7 @@ describe('Test for missing config props canvas warnings', () => { cy.interactWithConfigInputObject('parameters.repoName', 'test'); cy.closeStepConfigurationTab(); - cy.get('[data-id^="github"] g') + cy.get('[data-id^="camel-route|route.from.steps.1.to"] g') .find('span[data-warning="true"].pf-v5-c-icon') .should('have.attr', 'title', '2 required parameters are not yet configured: [ type,repoOwner ]'); }); @@ -30,7 +30,7 @@ describe('Test for missing config props canvas warnings', () => { cy.uploadFixture('flows/pipe/errorHandler.yaml'); cy.openDesignPage(); - cy.get('[data-id^="delay-action"] g') + cy.get('[data-id^="webhook-binding|delay-action"] g') .find('span[data-warning="true"].pf-v5-c-icon') .should('have.attr', 'title', '1 required parameter is not yet configured: [ milliseconds ]'); @@ -39,6 +39,8 @@ describe('Test for missing config props canvas warnings', () => { cy.interactWithConfigInputObject('milliseconds', '1000'); cy.closeStepConfigurationTab(); - cy.get('[data-id^="delay-action"] g').find('span[data-warning="true"].pf-v5-c-icon').should('not.exist'); + cy.get('[data-id^="webhook-binding|delay-action"] g') + .find('span[data-warning="true"].pf-v5-c-icon') + .should('not.exist'); }); }); diff --git a/packages/ui-tests/cypress/support/next-commands/design.ts b/packages/ui-tests/cypress/support/next-commands/design.ts index b061a7d43..ed39aea91 100644 --- a/packages/ui-tests/cypress/support/next-commands/design.ts +++ b/packages/ui-tests/cypress/support/next-commands/design.ts @@ -106,13 +106,13 @@ Cypress.Commands.add('checkNodeExist', (inputName, nodesCount) => { }); Cypress.Commands.add('checkEdgeExists', (sourceName: string, targetName: string) => { - const idPattern = sourceName + '-\\d+-to-' + targetName + '-\\d+'; + const idPattern = `${sourceName} >>> ${targetName}`; // Check if an element with the matching id exists cy.get('g').should(($elements) => { // Use Cypress commands to check if any element matches the id pattern const matchingElementExists = $elements.toArray().some((element) => { const dataId = Cypress.$(element).attr('data-id'); - return dataId && dataId.match(idPattern); + return dataId === idPattern; }); // Assert that at least one matching element exists expect(matchingElementExists).to.be.true; diff --git a/packages/ui/src/components/Visualization/Canvas/Canvas.tsx b/packages/ui/src/components/Visualization/Canvas/Canvas.tsx index 1f6c261ad..6f6af11b7 100644 --- a/packages/ui/src/components/Visualization/Canvas/Canvas.tsx +++ b/packages/ui/src/components/Visualization/Canvas/Canvas.tsx @@ -74,7 +74,7 @@ export const Canvas: FunctionComponent> = ({ enti entities.forEach((entity) => { if (visibleFlows[entity.id]) { - const { nodes: childNodes, edges: childEdges } = FlowService.getFlowDiagram(entity.toVizNode()); + const { nodes: childNodes, edges: childEdges } = FlowService.getFlowDiagram(entity.id, entity.toVizNode()); nodes.push(...childNodes); edges.push(...childEdges); } diff --git a/packages/ui/src/components/Visualization/Canvas/CanvasSideBar.test.tsx b/packages/ui/src/components/Visualization/Canvas/CanvasSideBar.test.tsx index 105c3f3f6..a019ede7c 100644 --- a/packages/ui/src/components/Visualization/Canvas/CanvasSideBar.test.tsx +++ b/packages/ui/src/components/Visualization/Canvas/CanvasSideBar.test.tsx @@ -16,7 +16,7 @@ describe('CanvasSideBar', () => { const camelResource = new CamelRouteResource(); camelResource.addNewEntity(EntityType.Route); const visualEntity = camelResource.getVisualEntities()[0]; - selectedNode = FlowService.getFlowDiagram(visualEntity.toVizNode()).nodes[0]; + selectedNode = FlowService.getFlowDiagram('test', visualEntity.toVizNode()).nodes[0]; Provider = TestProvidersWrapper({ camelResource }).Provider; }); diff --git a/packages/ui/src/components/Visualization/Canvas/Form/CanvasForm.test.tsx b/packages/ui/src/components/Visualization/Canvas/Form/CanvasForm.test.tsx index 6fc9a871a..e4a3ee407 100644 --- a/packages/ui/src/components/Visualization/Canvas/Form/CanvasForm.test.tsx +++ b/packages/ui/src/components/Visualization/Canvas/Form/CanvasForm.test.tsx @@ -52,7 +52,7 @@ describe('CanvasForm', () => { beforeEach(() => { camelRouteVisualEntity = new CamelRouteVisualEntity(camelRouteJson); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[2]; // choice }); @@ -145,7 +145,7 @@ describe('CanvasForm', () => { const flowId = camelRouteVisualEntity.id; const dispatchSpy = jest.fn(); const visualFlowsApi = new VisualFlowsApi(dispatchSpy); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[nodes.length - 1]; render( @@ -182,7 +182,7 @@ describe('CanvasForm', () => { const flowId = camelRouteVisualEntity.id; const dispatchSpy = jest.fn(); const visualFlowsApi = new VisualFlowsApi(dispatchSpy); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[nodes.length - 1]; render( @@ -220,7 +220,7 @@ describe('CanvasForm', () => { const newName = 'MyNewId'; const dispatchSpy = jest.fn(); const visualFlowsApi = new VisualFlowsApi(dispatchSpy); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[nodes.length - 1]; render( @@ -260,7 +260,7 @@ describe('CanvasForm', () => { const newName = 'MyNewName'; const dispatchSpy = jest.fn(); const visualFlowsApi = new VisualFlowsApi(dispatchSpy); - const { nodes } = FlowService.getFlowDiagram(kameletVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', kameletVisualEntity.toVizNode()); selectedNode = nodes[nodes.length - 1]; render( @@ -292,7 +292,7 @@ describe('CanvasForm', () => { describe('should show the User-updated field under the modified tab', () => { beforeEach(() => { camelRouteVisualEntity = new CamelRouteVisualEntity(camelRouteJson); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[0]; // timer }); @@ -629,7 +629,7 @@ describe('CanvasForm', () => { describe('should show the Required field under the required tab', () => { beforeEach(() => { camelRouteVisualEntity = new CamelRouteVisualEntity(camelRouteJson); - const { nodes } = FlowService.getFlowDiagram(camelRouteVisualEntity.toVizNode()); + const { nodes } = FlowService.getFlowDiagram('test', camelRouteVisualEntity.toVizNode()); selectedNode = nodes[0]; // timer }); diff --git a/packages/ui/src/components/Visualization/Canvas/Form/__snapshots__/CanvasForm.test.tsx.snap b/packages/ui/src/components/Visualization/Canvas/Form/__snapshots__/CanvasForm.test.tsx.snap index d41c19590..18b04f08f 100644 --- a/packages/ui/src/components/Visualization/Canvas/Form/__snapshots__/CanvasForm.test.tsx.snap +++ b/packages/ui/src/components/Visualization/Canvas/Form/__snapshots__/CanvasForm.test.tsx.snap @@ -23,7 +23,7 @@ exports[`CanvasForm should render 1`] = ` > icon