Skip to content

Commit eeea2bf

Browse files
committed
Added testing with Cypress
1 parent 2ebe447 commit eeea2bf

20 files changed

+1928
-51
lines changed

.github/workflows/test.yml

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
name: Tests
2+
3+
on:
4+
pull_request:
5+
6+
concurrency:
7+
group: "test-${{ github.event.pull_request.head.ref }}"
8+
cancel-in-progress: false
9+
10+
jobs:
11+
test:
12+
name: Tests
13+
runs-on: ubuntu-latest
14+
15+
steps:
16+
- name: Checkout code
17+
uses: actions/checkout@v2
18+
19+
- name: Install dependencies
20+
run: yarn install
21+
22+
- name: Run Quasar E2E tests
23+
run: yarn test:e2e:ci
24+
25+
- name: Upload Test Artifacts
26+
uses: actions/upload-artifact@v2
27+
with:
28+
name: test-artifacts
29+
path: |
30+
test/e2e/screenshots
31+
test/e2e/videos

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,6 @@ node_modules/
66
.vscode
77
.quasar
88
/dist/
9+
10+
.nyc_output
11+
coverage/

.nycrc

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"extends": "@quasar/quasar-app-extension-testing-e2e-cypress/nyc-config-preset"
3+
}

cypress.config.cjs

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
const registerCodeCoverageTasks = require('@cypress/code-coverage/task');
2+
const { injectQuasarDevServerConfig } = require('@quasar/quasar-app-extension-testing-e2e-cypress/cct-dev-server');
3+
const { defineConfig } = require('cypress');
4+
5+
module.exports = defineConfig({
6+
fixturesFolder: 'test/cypress/fixtures',
7+
screenshotsFolder: 'test/cypress/screenshots',
8+
videosFolder: 'test/cypress/videos',
9+
video: true,
10+
e2e: {
11+
setupNodeEvents(on, config) {
12+
registerCodeCoverageTasks(on, config);
13+
return config;
14+
},
15+
baseUrl: 'http://localhost:8080/',
16+
supportFile: 'test/cypress/support/e2e.js',
17+
specPattern: 'test/cypress/e2e/**/*.cy.{js,jsx,ts,tsx}',
18+
},
19+
component: {
20+
setupNodeEvents(on, config) {
21+
registerCodeCoverageTasks(on, config);
22+
return config;
23+
},
24+
supportFile: 'test/cypress/support/component.js',
25+
specPattern: 'src/**/*.cy.{js,jsx,ts,tsx}',
26+
indexHtmlFile: 'test/cypress/support/component-index.html',
27+
devServer: injectQuasarDevServerConfig(),
28+
},
29+
});

package.json

+8-3
Original file line numberDiff line numberDiff line change
@@ -8,14 +8,16 @@
88
"scripts": {
99
"lint": "eslint --ext .js,.vue ./",
1010
"format": "prettier --write \"**/*.{js,vue,scss,html,md,json}\" --ignore-path .gitignore",
11-
"test": "echo \"No test specified\" && exit 0",
11+
"test": "echo \"See package.json => scripts for available tests.\" && exit 0",
1212
"dev:spa": "quasar dev -m spa",
1313
"build:spa": "quasar build -m spa",
1414
"dev:ssr": "quasar dev -m ssr",
1515
"build:ssr": "quasar build -m ssr",
1616
"build:ssg": "quasar ssg generate",
1717
"dev:ssg": "quasar ssg dev",
18-
"serve:ssg": "quasar ssg serve dist/ssg"
18+
"serve:ssg": "quasar ssg serve dist/ssg",
19+
"test:e2e": "cross-env NODE_ENV=test start-test \"quasar dev\" http-get://127.0.0.1:8080 \"cypress open --e2e\"",
20+
"test:e2e:ci": "cross-env NODE_ENV=test start-test \"quasar dev\" http-get://127.0.0.1:8080 \"cypress run --e2e\""
1921
},
2022
"dependencies": {
2123
"@quasar/cli": "^2.3.0",
@@ -36,13 +38,16 @@
3638
"@intlify/vite-plugin-vue-i18n": "^3.3.1",
3739
"@modyfi/vite-plugin-yaml": "^1.1.0",
3840
"@quasar/app-vite": "^1.4.3",
41+
"@quasar/quasar-app-extension-testing-e2e-cypress": "^5.2.0",
3942
"autoprefixer": "^10.4.2",
4043
"eslint": "^8.11.0",
4144
"eslint-config-prettier": "^8.1.0",
4245
"eslint-plugin-vue": "^9.0.0",
4346
"postcss": "^8.4.14",
4447
"prettier": "^2.5.1",
45-
"quasar-app-extension-ssg": "^5.1.1"
48+
"quasar-app-extension-ssg": "^5.1.1",
49+
"cypress": "^13.6.2",
50+
"eslint-plugin-cypress": "^2.15.1"
4651
},
4752
"engines": {
4853
"node": "^20 || ^18 || ^16",

quasar.extensions.json

+6
Original file line numberDiff line numberDiff line change
@@ -3,5 +3,11 @@
33
"scripts": true,
44
"IDE": true,
55
"inlineCriticalCss": true
6+
},
7+
"@quasar/testing-e2e-cypress": {
8+
"port": 8080,
9+
"options": [
10+
"code-coverage"
11+
]
612
}
713
}

src/components/LanguageSwitcher.vue

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
<q-avatar size="1.5em" square v-html="selectedLanguage.iconHTML" />
99
</template>
1010
<template v-for="(lang) in languageOptions" :key="lang.value">
11-
<q-fab-action external-label label-position="top" color="primary" :to="{name: 'index', params: {lang: lang.value}}">
11+
<q-fab-action external-label :data-locale="lang.value" label-position="top" color="primary" :to="{name: 'index', params: {lang: lang.value}}">
1212
<template v-slot:default>
1313
<q-avatar v-html="lang.iconHTML" size="1.5em" square class="q-mr-sm" />
1414
{{ lang.label }}

src/components/OtherLinksModal.vue

+9-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,18 @@
11
<template>
2-
<q-dialog v-model="show" square :class="{'full-width': $q.screen.lt.sm}">
2+
<q-dialog v-model="show" square :class="{'full-width': $q.screen.lt.sm}" id="extra_links_modal">
33
<q-card dark>
44
<q-card-section class="row items-center">
55
<q-input autofocus dark dense filled flat square v-model="searchQuery" :placeholder="$t('form.filter')"
6-
clearable class="full-width"/>
6+
clearable class="full-width extra_links_modal__search_field" name="search" type="search" />
77
</q-card-section>
88

99
<q-scroll-area style="max-height: 98vh; height: 200px; margin-right: 1em;" dark>
10-
<q-card-section class="scroll q-pt-none" v-if="!otherLinksStore.isLoading">
10+
<q-card-section class="scroll q-pt-none extra_links_modal__search-results"
11+
:class="{
12+
'extra_links_modal__search-results--non-empty': searchResults.length > 0,
13+
'extra_links_modal__search-results--empty': searchResults.length === 0,
14+
}"
15+
v-if="!otherLinksStore.isLoading">
1116
<q-list v-if="searchResults.length > 0">
1217
<q-item v-for="(result, index) in searchResults" :key="index" dense dark>
1318
<q-item-section side>
@@ -24,7 +29,7 @@
2429
{{ $t('form.no_found') }}
2530
</div>
2631
</q-card-section>
27-
<q-spinner-dots class="absolute-center" size="md" v-else />
32+
<q-spinner-dots class="absolute-center extra_links_modal__loader" size="md" v-else />
2833
</q-scroll-area>
2934

3035
<q-card-actions align="right" style="margin-top: auto;">

src/layouts/MainLayout.vue

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@
22
<q-layout view="hHh lpR fFf">
33
<q-page-container>
44
<q-no-ssr>
5-
<background-canvas class="absolute-top-left background-canvas" v-if="isLoaded" />
6-
<language-switcher />
5+
<background-canvas class="absolute-top-left background-canvas" v-if="isLoaded" id="background" />
6+
<language-switcher id="language_switcher" />
77
</q-no-ssr>
88
<router-view v-if="isLoaded" />
99
</q-page-container>

src/pages/IndexPage.vue

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
<template>
22
<q-page class="flex flex-center">
3-
<q-card class="no-border" flat dark ref="main_card">
3+
<q-card class="no-border" flat dark ref="main_card" id="main_card">
44
<q-card-section>
55
<div class="text-h1 non-selectable first-line" @dblclick="reloadBackground">{{ mainConfig.header.first_line }}</div>
66
<div class="text-h5 non-selectable text-justify second-line">{{ mainConfig.header.second_line }}</div>

test/cypress/e2e/IndexPage.cy.js

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
// todo: make locales dynamically filled
2+
const locales = ['en-US', 'lt'];
3+
4+
locales.concat(['']).sort((a,b) => a.length - b.length).forEach(
5+
(locale) => {
6+
describe('IndexPage (/' + locale + ')', () => {
7+
beforeEach(() => {
8+
cy.visit('/' + locale);
9+
});
10+
11+
it('clicking on header text changes background image', () => {
12+
cy.get('#main_card .text-h1').dblclick();
13+
});
14+
15+
it('other links opening', () => {
16+
cy.get('#main_card .q-btn:last-child > .q-btn__content').click();
17+
cy.get('#extra_links_modal input[name="search"]').click();
18+
cy.get('#extra_links_modal input[name="search"]').type('git');
19+
cy.get('#extra_links_modal .extra_links_modal__search-results')
20+
.should('have.class', 'extra_links_modal__search-results--non-empty');
21+
cy.get('#extra_links_modal .extra_links_modal__search-results .q-item')
22+
.should('have.length', 2);
23+
24+
cy.get('#extra_links_modal input[name="search"]').type('git-not-found');
25+
cy.get('#extra_links_modal .extra_links_modal__search-results')
26+
.should('have.class', 'extra_links_modal__search-results--empty');
27+
28+
cy.get('#extra_links_modal input[name="search"]').type('{esc}');
29+
});
30+
31+
locales.forEach(
32+
(locale2) => {
33+
it(`switching language into ${locale2}`, () => {
34+
cy.get('#language_switcher .q-icon').click();
35+
cy.get(`#language_switcher [data-locale="${locale2}"]`).click();
36+
cy.url().should('match', new RegExp(`\/${locale2}$`));
37+
});
38+
}
39+
);
40+
41+
});
42+
}
43+
);
44+
45+
/*cy.visit('http://localhost:9100/');
46+
47+
cy.visit('http://localhost:9100/en-US');
48+
cy.get('#f_eaa58fbc-b613-49dd-8120-4e34ecfded13').click();;
49+
50+
cy.get('.q-btn--standard:nth-child(2) > .q-btn__content').click();
51+
cy.get('.q-page').click();
52+
cy.get('#f_740cf067-e05d-4f10-9978-db6691274f0c').type('git');
53+
cy.get('.q-pt-none').click();
54+
cy.get('.block').click();
55+
cy.get('.q-btn--fab').click();
56+
cy.get('.disabled:nth-child(1) > .q-btn__content').click();
57+
cy.get('.q-btn--outline:nth-child(2) > .q-btn__content').click();
58+
cy.get('#f_56f30064-a097-474f-bc3f-672e4c9f9679').click();
59+
cy.get('#f_56f30064-a097-474f-bc3f-672e4c9f9679').type('git');
60+
cy.get('.block').click();*/
61+
62+
63+
// ** The following code is an example to show you how to write some tests for your home page **
64+
//
65+
// describe('Home page tests', () => {
66+
// beforeEach(() => {
67+
// cy.visit('/');
68+
// });
69+
// it('has pretty background', () => {
70+
// cy.dataCy('landing-wrapper')
71+
// .should('have.css', 'background').and('match', /(".+(\/img\/background).+\.png)/);
72+
// });
73+
// it('has pretty logo', () => {
74+
// cy.dataCy('landing-wrapper img')
75+
// .should('have.class', 'logo-main')
76+
// .and('have.attr', 'src')
77+
// .and('match', /^(data:image\/svg\+xml).+/);
78+
// });
79+
// it('has very important information', () => {
80+
// cy.dataCy('instruction-wrapper')
81+
// .should('contain', 'SETUP INSTRUCTIONS')
82+
// .and('contain', 'Configure Authentication')
83+
// .and('contain', 'Database Configuration and CRUD operations')
84+
// .and('contain', 'Continuous Integration & Continuous Deployment CI/CD');
85+
// });
86+
// });

test/cypress/screenshots/.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
*
2+
!.gitignore

test/cypress/support/commands.js

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
// ***********************************************
2+
// This example commands.js shows you how to
3+
// create various custom commands and overwrite
4+
// existing commands.
5+
//
6+
// For more comprehensive examples of custom
7+
// commands please read more here:
8+
// https://on.cypress.io/custom-commands
9+
// ***********************************************
10+
//
11+
//
12+
// -- This is a parent command --
13+
// Cypress.Commands.add("login", (email, password) => { ... })
14+
//
15+
//
16+
// -- This is a child command --
17+
// Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... })
18+
//
19+
//
20+
// -- This is a dual command --
21+
// Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... })
22+
//
23+
//
24+
// -- This is will overwrite an existing command --
25+
// Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... })
26+
27+
// DO NOT REMOVE
28+
// Imports Quasar Cypress AE predefined commands
29+
import { registerCommands } from '@quasar/quasar-app-extension-testing-e2e-cypress'
30+
31+
registerCommands();
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
<!DOCTYPE html>
2+
<html>
3+
<head>
4+
<meta charset="utf-8">
5+
<meta http-equiv="X-UA-Compatible" content="IE=edge">
6+
<meta name="viewport" content="width=device-width,initial-scale=1.0">
7+
<title>Components App</title>
8+
</head>
9+
<body>
10+
<div data-cy-root></div>
11+
</body>
12+
</html>

test/cypress/support/component.js

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
// ***********************************************************
2+
// This example support/component.js is processed and
3+
// loaded automatically before your component test files.
4+
//
5+
// This is a great place to put global configuration and
6+
// behavior that modifies Cypress.
7+
//
8+
// You can change the location of this file or turn off
9+
// automatically serving support files with the
10+
// 'supportFile' configuration option.
11+
//
12+
// You can read more here:
13+
// https://on.cypress.io/configuration
14+
// ***********************************************************
15+
16+
import './commands'
17+
import '@cypress/code-coverage/support'
18+
19+
// Quasar styles
20+
import 'quasar/src/css/index.sass' // Or 'quasar/dist/quasar.prod.css' if no CSS preprocessor is installed
21+
// Change this if you have a different entrypoint for the main scss.
22+
import 'src/css/app.scss' // Or 'src/css/app.css' if no CSS preprocessor is installed
23+
// ICON SETS
24+
// If you use multiple or different icon-sets then the default, be sure to import them here.
25+
import 'quasar/dist/icon-set/material-icons.umd.prod'
26+
import '@quasar/extras/material-icons/material-icons.css'
27+
28+
import { installQuasarPlugin } from '@quasar/quasar-app-extension-testing-e2e-cypress'
29+
import { Dialog } from 'quasar'
30+
31+
// Since Cypress v10 we cannot import `config` directly from VTU as Cypress bundles its own version of it
32+
// See https://github.com/cypress-io/cypress/issues/22611
33+
import { VueTestUtils } from 'cypress/vue'
34+
35+
const { config } = VueTestUtils;
36+
37+
// Example to import i18n from boot and use as plugin
38+
// import { i18n } from 'src/boot/i18n';
39+
40+
// You can modify the global config here for all tests or pass in the configuration per test
41+
// For example use the actual i18n instance or mock it
42+
// config.global.plugins.push(i18n);
43+
config.global.mocks = {
44+
$t: () => '',
45+
};
46+
47+
// Overwrite the transition and transition-group stubs which are stubbed by test-utils by default.
48+
// We do want transitions to show when doing visual testing :)
49+
config.global.stubs = {};
50+
51+
installQuasarPlugin({ plugins: { Dialog } });

test/cypress/support/e2e.js

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
// ***********************************************************
2+
// This example support/e2e.js is processed and
3+
// loaded automatically before your e2e test files.
4+
//
5+
// This is a great place to put global configuration and
6+
// behavior that modifies Cypress.
7+
//
8+
// You can change the location of this file or turn off
9+
// automatically serving support files with the
10+
// 'supportFile' configuration option.
11+
//
12+
// You can read more here:
13+
// https://on.cypress.io/configuration
14+
// ***********************************************************
15+
16+
import './commands'
17+
import '@cypress/code-coverage/support'

test/cypress/videos/.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
*
2+
!.gitignore

0 commit comments

Comments
 (0)