import * as setupCredsModal from '../composables/modals/workflow-credential-setup-modal'; import * as formStep from '../composables/setup-template-form-step'; import { getSetupWorkflowCredentialsButton } from '../composables/setup-workflow-credentials-button'; import TestTemplate1 from '../fixtures/Test_Template_1.json'; import TestTemplate2 from '../fixtures/Test_Template_2.json'; import { clickUseWorkflowButtonByTitle, visitTemplateCollectionPage, testData, } from '../pages/template-collection'; import * as templateCredentialsSetupPage from '../pages/template-credential-setup'; import { WorkflowPage } from '../pages/workflow'; const workflowPage = new WorkflowPage(); const testTemplate = { id: 1205, data: TestTemplate1, }; const templateWithoutCredentials = { id: 1344, data: TestTemplate2, }; // NodeView uses beforeunload listener that will show a browser // native popup, which will block cypress from continuing / exiting. // This prevent the registration of the listener. Cypress.on('window:before:load', (win) => { const origAddEventListener = win.addEventListener; win.addEventListener = (eventName: string, listener: any, opts: any) => { if (eventName === 'beforeunload') { return; } return origAddEventListener.call(win, eventName, listener, opts); }; }); describe('Template credentials setup', () => { beforeEach(() => { cy.intercept( 'GET', `https://api.n8n.io/api/templates/workflows/${testTemplate.id}`, testTemplate.data, ).as('getTemplatePreview'); cy.intercept( 'GET', `https://api.n8n.io/api/workflows/templates/${testTemplate.id}`, testTemplate.data.workflow, ).as('getTemplate'); cy.overrideSettings({ templates: { enabled: true, host: 'https://api.n8n.io/api/' }, }); }); it('can be opened from template collection page', () => { visitTemplateCollectionPage(testData.ecommerceStarterPack); templateCredentialsSetupPage.enableTemplateCredentialSetupFeatureFlag(); clickUseWorkflowButtonByTitle('Promote new Shopify products'); templateCredentialsSetupPage.getters .title("Set up 'Promote new Shopify products' template") .should('be.visible'); }); it('has all the elements on page', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); templateCredentialsSetupPage.getters .title("Set up 'Promote new Shopify products' template") .should('be.visible'); templateCredentialsSetupPage.getters .infoCallout() .should( 'contain.text', 'You need 1x Shopify, 1x X (Formerly Twitter) and 1x Telegram account to setup this template', ); const expectedAppNames = ['1. Shopify', '2. X (Formerly Twitter)', '3. Telegram']; const expectedAppDescriptions = [ 'The credential you select will be used in the product created node of the workflow template.', 'The credential you select will be used in the Twitter node of the workflow template.', 'The credential you select will be used in the Telegram node of the workflow template.', ]; formStep.getFormStep().each(($el, index) => { formStep.getStepHeading($el).should('have.text', expectedAppNames[index]); formStep.getStepDescription($el).should('have.text', expectedAppDescriptions[index]); }); }); it('can skip template creation', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); templateCredentialsSetupPage.getters.skipLink().click(); workflowPage.getters.canvasNodes().should('have.length', 3); }); it('can create credentials and workflow from the template', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); // Continue button should be disabled if no credentials are created templateCredentialsSetupPage.getters.continueButton().should('be.disabled'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Shopify'); // Continue button should be enabled if at least one has been created templateCredentialsSetupPage.getters.continueButton().should('be.enabled'); templateCredentialsSetupPage.fillInDummyCredentialsForAppWithConfirm('X (Formerly Twitter)'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Telegram'); templateCredentialsSetupPage.finishCredentialSetup(); workflowPage.getters.canvasNodes().should('have.length', 3); // Focus the canvas so the copy to clipboard works workflowPage.getters.canvasNodes().eq(0).realClick(); workflowPage.actions.hitSelectAll(); workflowPage.actions.hitCopy(); cy.grantBrowserPermissions('clipboardReadWrite', 'clipboardSanitizedWrite'); // Check workflow JSON by copying it to clipboard cy.readClipboard().then((workflowJSON) => { const workflow = JSON.parse(workflowJSON); expect(workflow.meta).to.haveOwnProperty('templateId', testTemplate.id.toString()); expect(workflow.meta).not.to.haveOwnProperty('templateCredsSetupCompleted'); workflow.nodes.forEach((node: any) => { expect(Object.keys(node.credentials ?? {})).to.have.lengthOf(1); }); }); }); it('should work with a template that has no credentials (ADO-1603)', () => { const { id, data } = templateWithoutCredentials; cy.intercept('GET', `https://api.n8n.io/api/templates/workflows/${id}`, data); templateCredentialsSetupPage.visitTemplateCredentialSetupPage(id); const expectedAppNames = ['1. Email (IMAP)', '2. Nextcloud']; const expectedAppDescriptions = [ 'The credential you select will be used in the IMAP Email node of the workflow template.', 'The credential you select will be used in the Nextcloud node of the workflow template.', ]; formStep.getFormStep().each(($el, index) => { formStep.getStepHeading($el).should('have.text', expectedAppNames[index]); formStep.getStepDescription($el).should('have.text', expectedAppDescriptions[index]); }); templateCredentialsSetupPage.getters.continueButton().should('be.disabled'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Email (IMAP)'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Nextcloud'); templateCredentialsSetupPage.finishCredentialSetup(); workflowPage.getters.canvasNodes().should('have.length', 3); }); describe('Credential setup from workflow editor', { disableAutoLogin: true }, () => { beforeEach(() => { cy.resetDatabase(); cy.signinAsOwner(); }); it('should allow credential setup from workflow editor if user skips it during template setup', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); templateCredentialsSetupPage.getters.skipLink().click(); getSetupWorkflowCredentialsButton().should('be.visible'); }); it('should allow credential setup from workflow editor if user fills in credentials partially during template setup', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Shopify'); templateCredentialsSetupPage.finishCredentialSetup(); getSetupWorkflowCredentialsButton().should('be.visible'); }); it('should fill credentials from workflow editor', () => { templateCredentialsSetupPage.visitTemplateCredentialSetupPage(testTemplate.id); templateCredentialsSetupPage.getters.skipLink().click(); getSetupWorkflowCredentialsButton().click(); setupCredsModal.getWorkflowCredentialsModal().should('be.visible'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Shopify'); templateCredentialsSetupPage.fillInDummyCredentialsForAppWithConfirm('X (Formerly Twitter)'); templateCredentialsSetupPage.fillInDummyCredentialsForApp('Telegram'); setupCredsModal.closeModalFromContinueButton(); setupCredsModal.getWorkflowCredentialsModal().should('not.exist'); // Focus the canvas so the copy to clipboard works workflowPage.getters.canvasNodes().eq(0).realClick(); workflowPage.actions.hitSelectAll(); workflowPage.actions.hitCopy(); cy.grantBrowserPermissions('clipboardReadWrite', 'clipboardSanitizedWrite'); // Check workflow JSON by copying it to clipboard cy.readClipboard().then((workflowJSON) => { const workflow = JSON.parse(workflowJSON); workflow.nodes.forEach((node: any) => { expect(Object.keys(node.credentials ?? {})).to.have.lengthOf(1); }); }); getSetupWorkflowCredentialsButton().should('not.exist'); }); }); });