feat: Introduced cypress tests in erpnext
This commit is contained in:
parent
81522ec521
commit
74b8c99bc2
@ -147,11 +147,14 @@
|
|||||||
"Chart": true,
|
"Chart": true,
|
||||||
"Cypress": true,
|
"Cypress": true,
|
||||||
"cy": true,
|
"cy": true,
|
||||||
|
"describe": true,
|
||||||
|
"expect": true,
|
||||||
"it": true,
|
"it": true,
|
||||||
"context": true,
|
"context": true,
|
||||||
"before": true,
|
"before": true,
|
||||||
"beforeEach": true,
|
"beforeEach": true,
|
||||||
"onScan": true,
|
"onScan": true,
|
||||||
"extend_cscript": true
|
"extend_cscript": true
|
||||||
|
"localforage": true,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
11
cypress.json
Normal file
11
cypress.json
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"baseUrl": "http://test-develop:8001/",
|
||||||
|
"projectId": "92odwv",
|
||||||
|
"adminPassword": "admin",
|
||||||
|
"defaultCommandTimeout": 20000,
|
||||||
|
"pageLoadTimeout": 15000,
|
||||||
|
"retries": {
|
||||||
|
"runMode": 2,
|
||||||
|
"openMode": 2
|
||||||
|
}
|
||||||
|
}
|
5
cypress/fixtures/example.json
Normal file
5
cypress/fixtures/example.json
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"name": "Using fixtures to represent data",
|
||||||
|
"email": "hello@cypress.io",
|
||||||
|
"body": "Fixtures are a great way to mock data for responses to routes"
|
||||||
|
}
|
13
cypress/integration/test_customer.js
Normal file
13
cypress/integration/test_customer.js
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
|
||||||
|
context('Customer', () => {
|
||||||
|
before(() => {
|
||||||
|
cy.login();
|
||||||
|
});
|
||||||
|
it('Check Customer Group', () => {
|
||||||
|
cy.visit(`app/customer/`);
|
||||||
|
cy.get('.primary-action').click();
|
||||||
|
cy.wait(500);
|
||||||
|
cy.get('.custom-actions > .btn').click();
|
||||||
|
cy.get_field('customer_group', 'Link').should('have.value', 'All Customer Groups');
|
||||||
|
});
|
||||||
|
});
|
17
cypress/plugins/index.js
Normal file
17
cypress/plugins/index.js
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
// ***********************************************************
|
||||||
|
// This example plugins/index.js can be used to load plugins
|
||||||
|
//
|
||||||
|
// You can change the location of this file or turn off loading
|
||||||
|
// the plugins file with the 'pluginsFile' configuration option.
|
||||||
|
//
|
||||||
|
// You can read more here:
|
||||||
|
// https://on.cypress.io/plugins-guide
|
||||||
|
// ***********************************************************
|
||||||
|
|
||||||
|
// This function is called when a project is opened or re-opened (e.g. due to
|
||||||
|
// the project's config changing)
|
||||||
|
|
||||||
|
module.exports = () => {
|
||||||
|
// `on` is used to hook into various events Cypress emits
|
||||||
|
// `config` is the resolved Cypress config
|
||||||
|
};
|
326
cypress/support/commands.js
Normal file
326
cypress/support/commands.js
Normal file
@ -0,0 +1,326 @@
|
|||||||
|
import 'cypress-file-upload';
|
||||||
|
// ***********************************************
|
||||||
|
// This example commands.js shows you how to
|
||||||
|
// create various custom commands and overwrite
|
||||||
|
// existing commands.
|
||||||
|
//
|
||||||
|
// For more comprehensive examples of custom
|
||||||
|
// commands please read more here:
|
||||||
|
// https://on.cypress.io/custom-commands
|
||||||
|
// ***********************************************
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// -- This is a parent command --
|
||||||
|
// Cypress.Commands.add("login", (email, password) => { ... });
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// -- This is a child command --
|
||||||
|
// Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... });
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// -- This is a dual command --
|
||||||
|
// Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... });
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// -- This is will overwrite an existing command --
|
||||||
|
// Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... });
|
||||||
|
Cypress.Commands.add('login', (email, password) => {
|
||||||
|
if (!email) {
|
||||||
|
email = 'Administrator';
|
||||||
|
}
|
||||||
|
if (!password) {
|
||||||
|
password = Cypress.config('adminPassword');
|
||||||
|
}
|
||||||
|
cy.request({
|
||||||
|
url: '/api/method/login',
|
||||||
|
method: 'POST',
|
||||||
|
body: {
|
||||||
|
usr: email,
|
||||||
|
pwd: password
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('call', (method, args) => {
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
url: `/api/method/${method}`,
|
||||||
|
method: 'POST',
|
||||||
|
body: args,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
expect(res.status).eq(200);
|
||||||
|
return res.body;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('get_list', (doctype, fields = [], filters = []) => {
|
||||||
|
filters = JSON.stringify(filters);
|
||||||
|
fields = JSON.stringify(fields);
|
||||||
|
let url = `/api/resource/${doctype}?fields=${fields}&filters=${filters}`;
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
method: 'GET',
|
||||||
|
url,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
expect(res.status).eq(200);
|
||||||
|
return res.body;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('get_doc', (doctype, name) => {
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
method: 'GET',
|
||||||
|
url: `/api/resource/${doctype}/${name}`,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
expect(res.status).eq(200);
|
||||||
|
return res.body;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('insert_doc', (doctype, args, ignore_duplicate) => {
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
method: 'POST',
|
||||||
|
url: `/api/resource/${doctype}`,
|
||||||
|
body: args,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
},
|
||||||
|
failOnStatusCode: !ignore_duplicate
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
let status_codes = [200];
|
||||||
|
if (ignore_duplicate) {
|
||||||
|
status_codes.push(409);
|
||||||
|
}
|
||||||
|
expect(res.status).to.be.oneOf(status_codes);
|
||||||
|
return res.body;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('remove_doc', (doctype, name) => {
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
method: 'DELETE',
|
||||||
|
url: `/api/resource/${doctype}/${name}`,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
expect(res.status).eq(202);
|
||||||
|
return res.body;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('create_records', doc => {
|
||||||
|
return cy
|
||||||
|
.call('frappe.tests.ui_test_helpers.create_if_not_exists', {doc})
|
||||||
|
.then(r => r.message);
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('set_value', (doctype, name, obj) => {
|
||||||
|
return cy.call('frappe.client.set_value', {
|
||||||
|
doctype,
|
||||||
|
name,
|
||||||
|
fieldname: obj
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('fill_field', (fieldname, value, fieldtype = 'Data') => {
|
||||||
|
cy.get_field(fieldname, fieldtype).as('input');
|
||||||
|
|
||||||
|
if (['Date', 'Time', 'Datetime'].includes(fieldtype)) {
|
||||||
|
cy.get('@input').click().wait(200);
|
||||||
|
cy.get('.datepickers-container .datepicker.active').should('exist');
|
||||||
|
}
|
||||||
|
if (fieldtype === 'Time') {
|
||||||
|
cy.get('@input').clear().wait(200);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fieldtype === 'Select') {
|
||||||
|
cy.get('@input').select(value);
|
||||||
|
} else {
|
||||||
|
cy.get('@input').type(value, {waitForAnimations: false, force: true});
|
||||||
|
}
|
||||||
|
return cy.get('@input');
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('get_field', (fieldname, fieldtype = 'Data') => {
|
||||||
|
let selector = `.form-control[data-fieldname="${fieldname}"]`;
|
||||||
|
|
||||||
|
if (fieldtype === 'Text Editor') {
|
||||||
|
selector = `[data-fieldname="${fieldname}"] .ql-editor[contenteditable=true]`;
|
||||||
|
}
|
||||||
|
if (fieldtype === 'Code') {
|
||||||
|
selector = `[data-fieldname="${fieldname}"] .ace_text-input`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return cy.get(selector);
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('fill_table_field', (tablefieldname, row_idx, fieldname, value, fieldtype = 'Data') => {
|
||||||
|
cy.get_table_field(tablefieldname, row_idx, fieldname, fieldtype).as('input');
|
||||||
|
|
||||||
|
if (['Date', 'Time', 'Datetime'].includes(fieldtype)) {
|
||||||
|
cy.get('@input').click().wait(200);
|
||||||
|
cy.get('.datepickers-container .datepicker.active').should('exist');
|
||||||
|
}
|
||||||
|
if (fieldtype === 'Time') {
|
||||||
|
cy.get('@input').clear().wait(200);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fieldtype === 'Select') {
|
||||||
|
cy.get('@input').select(value);
|
||||||
|
} else {
|
||||||
|
cy.get('@input').type(value, {waitForAnimations: false, force: true});
|
||||||
|
}
|
||||||
|
return cy.get('@input');
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('get_table_field', (tablefieldname, row_idx, fieldname, fieldtype = 'Data') => {
|
||||||
|
let selector = `.frappe-control[data-fieldname="${tablefieldname}"]`;
|
||||||
|
selector += ` [data-idx="${row_idx}"]`;
|
||||||
|
selector += ` .form-in-grid`;
|
||||||
|
|
||||||
|
if (fieldtype === 'Text Editor') {
|
||||||
|
selector += ` [data-fieldname="${fieldname}"] .ql-editor[contenteditable=true]`;
|
||||||
|
} else if (fieldtype === 'Code') {
|
||||||
|
selector += ` [data-fieldname="${fieldname}"] .ace_text-input`;
|
||||||
|
} else {
|
||||||
|
selector += ` .form-control[data-fieldname="${fieldname}"]`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return cy.get(selector);
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('awesomebar', text => {
|
||||||
|
cy.get('#navbar-search').type(`${text}{downarrow}{enter}`, {delay: 100});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('new_form', doctype => {
|
||||||
|
let dt_in_route = doctype.toLowerCase().replace(/ /g, '-');
|
||||||
|
cy.visit(`/app/${dt_in_route}/new`);
|
||||||
|
cy.get('body').should('have.attr', 'data-route', `Form/${doctype}/new-${dt_in_route}-1`);
|
||||||
|
cy.get('body').should('have.attr', 'data-ajax-state', 'complete');
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('go_to_list', doctype => {
|
||||||
|
cy.visit(`/app/list/${doctype}/list`);
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('clear_cache', () => {
|
||||||
|
cy.window()
|
||||||
|
.its('frappe')
|
||||||
|
.then(frappe => {
|
||||||
|
frappe.ui.toolbar.clear_cache();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('dialog', opts => {
|
||||||
|
return cy.window().then(win => {
|
||||||
|
var d = new win.frappe.ui.Dialog(opts);
|
||||||
|
d.show();
|
||||||
|
return d;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('get_open_dialog', () => {
|
||||||
|
return cy.get('.modal:visible').last();
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('hide_dialog', () => {
|
||||||
|
cy.wait(300);
|
||||||
|
cy.get_open_dialog().find('.btn-modal-close').click();
|
||||||
|
cy.get('.modal:visible').should('not.exist');
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('insert_doc', (doctype, args, ignore_duplicate) => {
|
||||||
|
return cy
|
||||||
|
.window()
|
||||||
|
.its('frappe.csrf_token')
|
||||||
|
.then(csrf_token => {
|
||||||
|
return cy
|
||||||
|
.request({
|
||||||
|
method: 'POST',
|
||||||
|
url: `/api/resource/${doctype}`,
|
||||||
|
body: args,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'X-Frappe-CSRF-Token': csrf_token
|
||||||
|
},
|
||||||
|
failOnStatusCode: !ignore_duplicate
|
||||||
|
})
|
||||||
|
.then(res => {
|
||||||
|
let status_codes = [200];
|
||||||
|
if (ignore_duplicate) {
|
||||||
|
status_codes.push(409);
|
||||||
|
}
|
||||||
|
expect(res.status).to.be.oneOf(status_codes);
|
||||||
|
return res.body.data;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('add_filter', () => {
|
||||||
|
cy.get('.filter-section .filter-button').click();
|
||||||
|
cy.wait(300);
|
||||||
|
cy.get('.filter-popover').should('exist');
|
||||||
|
});
|
||||||
|
|
||||||
|
Cypress.Commands.add('clear_filters', () => {
|
||||||
|
cy.get('.filter-section .filter-button').click();
|
||||||
|
cy.wait(300);
|
||||||
|
cy.get('.filter-popover').should('exist');
|
||||||
|
cy.get('.filter-popover').find('.clear-filters').click();
|
||||||
|
cy.get('.filter-section .filter-button').click();
|
||||||
|
cy.window().its('cur_list').then(cur_list => {
|
||||||
|
cur_list && cur_list.filter_area && cur_list.filter_area.clear();
|
||||||
|
});
|
||||||
|
});
|
25
cypress/support/index.js
Normal file
25
cypress/support/index.js
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
// ***********************************************************
|
||||||
|
// This example support/index.js is processed and
|
||||||
|
// loaded automatically before your test files.
|
||||||
|
//
|
||||||
|
// This is a great place to put global configuration and
|
||||||
|
// behavior that modifies Cypress.
|
||||||
|
//
|
||||||
|
// You can change the location of this file or turn off
|
||||||
|
// automatically serving support files with the
|
||||||
|
// 'supportFile' configuration option.
|
||||||
|
//
|
||||||
|
// You can read more here:
|
||||||
|
// https://on.cypress.io/configuration
|
||||||
|
// ***********************************************************
|
||||||
|
|
||||||
|
// Import commands.js using ES2015 syntax:
|
||||||
|
import './commands';
|
||||||
|
|
||||||
|
|
||||||
|
// Alternatively you can use CommonJS syntax:
|
||||||
|
// require('./commands')
|
||||||
|
|
||||||
|
Cypress.Cookies.defaults({
|
||||||
|
preserve: 'sid'
|
||||||
|
});
|
12
cypress/tsconfig.json
Normal file
12
cypress/tsconfig.json
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"allowJs": true,
|
||||||
|
"baseUrl": "../node_modules",
|
||||||
|
"types": [
|
||||||
|
"cypress"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"include": [
|
||||||
|
"**/*.*"
|
||||||
|
]
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user