refactor: extend commands from frappe
diff --git a/.github/workflows/ui-tests.yml b/.github/workflows/ui-tests.yml
index b187dff..4bc55da 100644
--- a/.github/workflows/ui-tests.yml
+++ b/.github/workflows/ui-tests.yml
@@ -91,6 +91,9 @@
       - name: Site Setup
         run: cd ~/frappe-bench/ && bench --site test_site execute erpnext.setup.utils.before_tests
 
+      - name: cypress pre-requisites
+        run: cd ~/frappe-bench/apps/frappe && yarn add cypress-file-upload@^5 --no-lockfile
+
 
       - name: Build Assets
         run: cd ~/frappe-bench/ && bench build
diff --git a/cypress/support/commands.js b/cypress/support/commands.js
index 1964b96..7929a2e 100644
--- a/cypress/support/commands.js
+++ b/cypress/support/commands.js
@@ -1,4 +1,3 @@
-import 'cypress-file-upload';
 // ***********************************************
 // This example commands.js shows you how to
 // create various custom commands and overwrite
@@ -24,303 +23,3 @@
 //
 // -- 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();
-	});
-});
diff --git a/cypress/support/index.js b/cypress/support/index.js
index 1bee72d..72070cc 100644
--- a/cypress/support/index.js
+++ b/cypress/support/index.js
@@ -15,6 +15,7 @@
 
 // Import commands.js using ES2015 syntax:
 import './commands';
+import '../../../frappe/cypress/support/commands' // eslint-disable-line
 
 
 // Alternatively you can use CommonJS syntax:
@@ -22,4 +23,4 @@
 
 Cypress.Cookies.defaults({
 	preserve: 'sid'
-});
\ No newline at end of file
+});