mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-18 08:00:48 -08:00
247 lines
8.1 KiB
TypeScript
247 lines
8.1 KiB
TypeScript
import OnboardingWorkflow from '../fixtures/Onboarding_workflow.json';
|
|
import WorkflowTemplate from '../fixtures/Workflow_template_write_http_query.json';
|
|
import { MainSidebar } from '../pages/sidebar/main-sidebar';
|
|
import { TemplatesPage } from '../pages/templates';
|
|
import { WorkflowPage } from '../pages/workflow';
|
|
import { WorkflowsPage } from '../pages/workflows';
|
|
|
|
const templatesPage = new TemplatesPage();
|
|
const workflowPage = new WorkflowPage();
|
|
const workflowsPage = new WorkflowsPage();
|
|
const mainSidebar = new MainSidebar();
|
|
|
|
describe('Workflow templates', () => {
|
|
const mockTemplateHost = (host: string) => {
|
|
cy.overrideSettings({
|
|
templates: { enabled: true, host },
|
|
});
|
|
};
|
|
|
|
describe('For api.n8n.io', () => {
|
|
beforeEach(() => {
|
|
mockTemplateHost('https://api.n8n.io/api/');
|
|
});
|
|
|
|
it('Opens website when clicking templates sidebar link', () => {
|
|
cy.visit(workflowsPage.url);
|
|
mainSidebar.getters.templates().should('be.visible');
|
|
// Templates should be a link to the website
|
|
mainSidebar.getters
|
|
.templates()
|
|
.parent('a')
|
|
.should('have.attr', 'href')
|
|
.and('include', 'https://n8n.io/workflows');
|
|
// Link should contain instance address and n8n version
|
|
mainSidebar.getters
|
|
.templates()
|
|
.parent('a')
|
|
.then(($a) => {
|
|
const href = $a.attr('href');
|
|
const params = new URLSearchParams(href);
|
|
// Link should have all mandatory parameters expected on the website
|
|
expect(decodeURIComponent(`${params.get('utm_instance')}`)).to.include(
|
|
window.location.origin,
|
|
);
|
|
expect(params.get('utm_n8n_version')).to.match(/[0-9]+\.[0-9]+\.[0-9]+/);
|
|
expect(params.get('utm_awc')).to.match(/[0-9]+/);
|
|
});
|
|
mainSidebar.getters.templates().parent('a').should('have.attr', 'target', '_blank');
|
|
});
|
|
|
|
it('Redirects to website when visiting templates page directly', () => {
|
|
cy.intercept(
|
|
{
|
|
hostname: 'n8n.io',
|
|
pathname: '/workflows',
|
|
},
|
|
'Mock Template Page',
|
|
).as('templatesPage');
|
|
|
|
cy.visit(templatesPage.url);
|
|
|
|
cy.wait('@templatesPage');
|
|
});
|
|
});
|
|
|
|
describe('For a custom template host', () => {
|
|
const hostname = 'random.domain';
|
|
const categories = [
|
|
{ id: 1, name: 'Engineering' },
|
|
{ id: 2, name: 'Finance' },
|
|
{ id: 3, name: 'Sales' },
|
|
];
|
|
const collections = [
|
|
{
|
|
id: 1,
|
|
name: 'Test Collection',
|
|
workflows: [{ id: 1 }],
|
|
nodes: [],
|
|
},
|
|
];
|
|
|
|
beforeEach(() => {
|
|
cy.intercept({ hostname, pathname: '/api/health' }, { status: 'OK' });
|
|
cy.intercept({ hostname, pathname: '/api/templates/categories' }, { categories });
|
|
cy.intercept(
|
|
{ hostname, pathname: '/api/templates/collections', query: { category: '**' } },
|
|
(req) => {
|
|
req.reply({ collections: req.query['category[]'] === '3' ? [] : collections });
|
|
},
|
|
);
|
|
cy.intercept(
|
|
{ hostname, pathname: '/api/templates/search', query: { category: '**' } },
|
|
(req) => {
|
|
const fixture =
|
|
req.query.category === 'Sales'
|
|
? 'templates_search/sales_templates_search_response.json'
|
|
: 'templates_search/all_templates_search_response.json';
|
|
req.reply({ statusCode: 200, fixture });
|
|
},
|
|
);
|
|
|
|
cy.intercept(
|
|
{ hostname, pathname: '/api/workflows/templates/1' },
|
|
{
|
|
statusCode: 200,
|
|
body: {
|
|
id: 1,
|
|
name: OnboardingWorkflow.name,
|
|
workflow: OnboardingWorkflow,
|
|
},
|
|
},
|
|
).as('getTemplate');
|
|
|
|
cy.intercept(
|
|
{ hostname, pathname: '/api/templates/workflows/1' },
|
|
{
|
|
statusCode: 200,
|
|
body: WorkflowTemplate,
|
|
},
|
|
).as('getTemplatePreview');
|
|
|
|
mockTemplateHost(`https://${hostname}/api`);
|
|
});
|
|
|
|
it('can open onboarding flow', () => {
|
|
templatesPage.actions.openOnboardingFlow();
|
|
cy.url().should('match', /.*\/workflow\/.*?onboardingId=1$/);
|
|
|
|
workflowPage.actions.shouldHaveWorkflowName('Demo: ' + OnboardingWorkflow.name);
|
|
workflowPage.getters.canvasNodes().should('have.length', 4);
|
|
workflowPage.getters.stickies().should('have.length', 1);
|
|
workflowPage.getters.canvasNodes().first().should('have.descendants', '.node-pin-data-icon');
|
|
});
|
|
|
|
it('can import template', () => {
|
|
templatesPage.actions.importTemplate();
|
|
cy.url().should('include', '/workflow/new?templateId=1');
|
|
|
|
workflowPage.getters.canvasNodes().should('have.length', 4);
|
|
workflowPage.getters.stickies().should('have.length', 1);
|
|
workflowPage.actions.shouldHaveWorkflowName(OnboardingWorkflow.name);
|
|
});
|
|
|
|
it('should save template id with the workflow', () => {
|
|
templatesPage.actions.importTemplate();
|
|
|
|
cy.visit(templatesPage.url);
|
|
cy.get('.el-skeleton.n8n-loading').should('not.exist');
|
|
templatesPage.getters.firstTemplateCard().should('exist');
|
|
templatesPage.getters.templatesLoadingContainer().should('not.exist');
|
|
templatesPage.getters.firstTemplateCard().click();
|
|
cy.url().should('include', '/templates/1');
|
|
cy.wait('@getTemplatePreview');
|
|
|
|
templatesPage.getters.useTemplateButton().click();
|
|
cy.url().should('include', '/workflow/new');
|
|
workflowPage.actions.saveWorkflowOnButtonClick();
|
|
|
|
workflowPage.actions.hitSelectAll();
|
|
workflowPage.actions.hitCopy();
|
|
|
|
cy.grantBrowserPermissions('clipboardReadWrite', 'clipboardSanitizedWrite');
|
|
// Check workflow JSON by copying it to clipboard
|
|
cy.readClipboard().then((workflowJSON) => {
|
|
expect(workflowJSON).to.contain('"templateId": "1"');
|
|
});
|
|
});
|
|
|
|
it('can open template with images and hides workflow screenshots', () => {
|
|
cy.visit(`${templatesPage.url}/1`);
|
|
cy.wait('@getTemplatePreview');
|
|
|
|
templatesPage.getters.description().find('img').should('have.length', 1);
|
|
});
|
|
|
|
it('renders search elements correctly', () => {
|
|
cy.visit(templatesPage.url);
|
|
templatesPage.getters.searchInput().should('exist');
|
|
templatesPage.getters.allCategoriesFilter().should('exist');
|
|
templatesPage.getters.categoryFilters().should('have.length.greaterThan', 1);
|
|
templatesPage.getters.templateCards().should('have.length.greaterThan', 0);
|
|
});
|
|
|
|
it('can filter templates by category', () => {
|
|
cy.visit(templatesPage.url);
|
|
templatesPage.getters.templatesLoadingContainer().should('not.exist');
|
|
templatesPage.getters.categoryFilter('sales').should('exist');
|
|
let initialTemplateCount = 0;
|
|
let initialCollectionCount = 0;
|
|
|
|
templatesPage.getters.templateCountLabel().then(($el) => {
|
|
initialTemplateCount = parseInt($el.text().replace(/\D/g, ''), 10);
|
|
templatesPage.getters.collectionCountLabel().then(($el1) => {
|
|
initialCollectionCount = parseInt($el1.text().replace(/\D/g, ''), 10);
|
|
|
|
templatesPage.getters.categoryFilter('sales').click();
|
|
templatesPage.getters.templatesLoadingContainer().should('not.exist');
|
|
|
|
// Should have less templates and collections after selecting a category
|
|
templatesPage.getters.templateCountLabel().should(($el2) => {
|
|
expect(parseInt($el2.text().replace(/\D/g, ''), 10)).to.be.lessThan(
|
|
initialTemplateCount,
|
|
);
|
|
});
|
|
templatesPage.getters.collectionCountLabel().should(($el2) => {
|
|
expect(parseInt($el2.text().replace(/\D/g, ''), 10)).to.be.lessThan(
|
|
initialCollectionCount,
|
|
);
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should preserve search query in URL', () => {
|
|
cy.visit(templatesPage.url);
|
|
templatesPage.getters.templatesLoadingContainer().should('not.exist');
|
|
templatesPage.getters.categoryFilter('sales').should('exist');
|
|
templatesPage.getters.categoryFilter('sales').click();
|
|
templatesPage.getters.searchInput().type('auto');
|
|
|
|
cy.url().should('include', '?categories=');
|
|
cy.url().should('include', '&search=');
|
|
|
|
cy.reload();
|
|
|
|
// Should preserve search query in URL
|
|
cy.url().should('include', '?categories=');
|
|
cy.url().should('include', '&search=');
|
|
|
|
// Sales category should still be selected
|
|
templatesPage.getters
|
|
.categoryFilter('sales')
|
|
.find('label')
|
|
.should('have.class', 'is-checked');
|
|
// Search input should still have the search query
|
|
templatesPage.getters.searchInput().should('have.value', 'auto');
|
|
// Sales checkbox should be pushed to the top
|
|
templatesPage.getters
|
|
.categoryFilters()
|
|
.eq(1)
|
|
.then(($el) => {
|
|
expect($el.text()).to.equal('Sales');
|
|
});
|
|
});
|
|
});
|
|
});
|