import { getManualChatModal } from './modals/chat-modal'; import { clickGetBackToCanvas, getParameterInputByName } from './ndv'; import { ROUTES } from '../constants'; /** * Types */ export type EndpointType = | 'ai_chain' | 'ai_document' | 'ai_embedding' | 'ai_languageModel' | 'ai_memory' | 'ai_outputParser' | 'ai_tool' | 'ai_retriever' | 'ai_textSplitter' | 'ai_vectorRetriever' | 'ai_vectorStore'; /** * Getters */ export function getAddInputEndpointByType(nodeName: string, endpointType: EndpointType) { return cy.get( `.add-input-endpoint[data-jtk-scope-${endpointType}][data-endpoint-name="${nodeName}"]`, ); } export function getNodeCreatorItems() { return cy.getByTestId('item-iterator-item'); } export function getExecuteWorkflowButton() { return cy.getByTestId('execute-workflow-button'); } export function getManualChatButton() { return cy.getByTestId('workflow-chat-button'); } export function getNodes() { return cy.getByTestId('canvas-node'); } export function getNodeByName(name: string) { return cy.ifCanvasVersion( () => cy.getByTestId('canvas-node').filter(`[data-name="${name}"]`).eq(0), () => cy.getByTestId('canvas-node').filter(`[data-node-name="${name}"]`).eq(0), ); } export function getWorkflowHistoryCloseButton() { return cy.getByTestId('workflow-history-close-button'); } export function disableNode(name: string) { const target = getNodeByName(name); target.rightclick(name ? 'center' : 'topLeft', { force: true }); cy.getByTestId('context-menu-item-toggle_activation').click(); } export function getConnectionBySourceAndTarget(source: string, target: string) { return cy .get('.jtk-connector') .filter(`[data-source-node="${source}"][data-target-node="${target}"]`) .eq(0); } export function getNodeCreatorSearchBar() { return cy.getByTestId('node-creator-search-bar'); } export function getNodeCreatorPlusButton() { return cy.getByTestId('node-creator-plus-button'); } export function getCanvasNodes() { return cy.ifCanvasVersion( () => cy.getByTestId('canvas-node'), () => cy.getByTestId('canvas-node').not('[data-node-type="n8n-nodes-internal.addNodes"]'), ); } export function getCanvasNodeByName(nodeName: string) { return getCanvasNodes().filter(`:contains(${nodeName})`); } export function getSaveButton() { return cy.getByTestId('workflow-save-button'); } export function getZoomToFitButton() { return cy.getByTestId('zoom-to-fit'); } /** * Actions */ export function addNodeToCanvas( nodeDisplayName: string, plusButtonClick = true, preventNdvClose?: boolean, action?: string, ) { if (plusButtonClick) { getNodeCreatorPlusButton().click(); } getNodeCreatorSearchBar().type(nodeDisplayName); getNodeCreatorSearchBar().type('{enter}'); cy.wait(500); cy.get('body').then((body) => { if (body.find('[data-test-id=node-creator]').length > 0) { if (action) { cy.contains(action).click(); } else { // Select the first action cy.get('[data-keyboard-nav-type="action"]').eq(0).click(); } } }); if (!preventNdvClose) cy.get('body').type('{esc}'); } export function navigateToNewWorkflowPage(preventNodeViewUnload = true) { cy.visit(ROUTES.NEW_WORKFLOW_PAGE); cy.waitForLoad(); cy.window().then((win) => { win.preventNodeViewBeforeUnload = preventNodeViewUnload; }); } function connectNodeToParent( nodeName: string, endpointType: EndpointType, parentNodeName: string, exactMatch = false, ) { getAddInputEndpointByType(parentNodeName, endpointType).click({ force: true }); if (exactMatch) { getNodeCreatorItems() .contains(new RegExp('^' + nodeName + '$', 'g')) .click(); } else { getNodeCreatorItems().contains(nodeName).click(); } } export function addSupplementalNodeToParent( nodeName: string, endpointType: EndpointType, parentNodeName: string, exactMatch = false, ) { connectNodeToParent(nodeName, endpointType, parentNodeName, exactMatch); getConnectionBySourceAndTarget(parentNodeName, nodeName).should('exist'); } export function addLanguageModelNodeToParent( nodeName: string, parentNodeName: string, exactMatch = false, ) { addSupplementalNodeToParent(nodeName, 'ai_languageModel', parentNodeName, exactMatch); } export function addMemoryNodeToParent(nodeName: string, parentNodeName: string) { addSupplementalNodeToParent(nodeName, 'ai_memory', parentNodeName); } export function addToolNodeToParent(nodeName: string, parentNodeName: string) { addSupplementalNodeToParent(nodeName, 'ai_tool', parentNodeName); } export function addVectorStoreToolToParent(nodeName: string, parentNodeName: string) { connectNodeToParent(nodeName, 'ai_tool', parentNodeName, false); getParameterInputByName('mode') .find('input') .should('have.value', 'Retrieve Documents (As Tool for AI Agent)'); clickGetBackToCanvas(); getConnectionBySourceAndTarget(nodeName, parentNodeName).should('exist'); } export function addOutputParserNodeToParent(nodeName: string, parentNodeName: string) { addSupplementalNodeToParent(nodeName, 'ai_outputParser', parentNodeName); } export function addVectorStoreNodeToParent(nodeName: string, parentNodeName: string) { addSupplementalNodeToParent(nodeName, 'ai_vectorStore', parentNodeName); } export function addRetrieverNodeToParent(nodeName: string, parentNodeName: string) { addSupplementalNodeToParent(nodeName, 'ai_retriever', parentNodeName); } export function clickExecuteWorkflowButton() { getExecuteWorkflowButton().click(); } export function clickManualChatButton() { getManualChatButton().click(); getManualChatModal().should('be.visible'); } export function openNode(nodeName: string) { getNodeByName(nodeName).dblclick(); } export function saveWorkflowOnButtonClick() { cy.intercept('POST', '/rest/workflows').as('createWorkflow'); getSaveButton().should('contain', 'Save'); getSaveButton().click(); getSaveButton().should('contain', 'Saved'); cy.url().should('not.have.string', '/new'); } export function pasteWorkflow(workflow: object) { cy.get('body').paste(JSON.stringify(workflow)); } export function clickZoomToFit() { getZoomToFitButton().click(); } export function deleteNode(name: string) { getCanvasNodeByName(name).first().click(); cy.get('body').type('{del}'); }