mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-12 05:17:28 -08:00
76dded4bea
Co-authored-by: Oleg Ivaniv <me@olegivaniv.com>
225 lines
5.9 KiB
TypeScript
225 lines
5.9 KiB
TypeScript
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.getByTestId('canvas-node').filter(`[data-name="${name}"]`).eq(0);
|
|
}
|
|
|
|
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}');
|
|
}
|