Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/keepassxreboot/keepassxc-browser.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'tests/tests.js')
-rw-r--r--tests/tests.js154
1 files changed, 154 insertions, 0 deletions
diff --git a/tests/tests.js b/tests/tests.js
new file mode 100644
index 0000000..ee6144a
--- /dev/null
+++ b/tests/tests.js
@@ -0,0 +1,154 @@
+'use strict';
+
+const Tests = {
+ GENERAL: '#general-results',
+ INPUT_FIELDS: '#input-field-results',
+ TOTP_FIELDS: '#totp-field-results',
+ SEARCH_FIELDS: '#search-field-results',
+ PASSWORD_CHANGE: '#password-change-results',
+};
+
+function createResult(card, res, text) {
+ const icon = kpxcUI.createElement('i', res ? 'fa fa-check' : 'fa fa-close');
+ const span = kpxcUI.createElement('span', '', '', text);
+ const br = document.createElement('br');
+
+ document.querySelector(card).appendMultiple(icon, span, br);
+}
+
+// General (global.js)
+async function testGeneral() {
+ const testCard = Tests.GENERAL;
+
+ // General
+ kpxcAssert(trimURL('https://test.com/path_to_somwhere?login=username'), 'https://test.com/path_to_somwhere', testCard, 'trimURL()');
+ assertRegex(slashNeededForUrl('https://test.com'), true, testCard, 'slashNeededForUrl()');
+ assertRegex(slashNeededForUrl('https://test.com/'), false, testCard, 'slashNeededForUrl()');
+
+ // URL matching (URL in Site Preferences, page URL, expected result).
+ // Consider using slighly different URL's for the tests cases.
+ const matches = [
+ [ 'https://example.com/*', 'https://example.com/login_page', true ],
+ [ 'https://example.com/*', 'https://example2.com/login_page', false ],
+ [ 'https://example.com/*', 'https://subdomain.example.com/login_page', false ],
+ [ 'https://*.example.com/*', 'https://example.com/login_page', true ],
+ [ 'https://*.example.com/*', 'https://test.example.com/login_page', true ],
+ [ 'https://test.example.com/*', 'https://subdomain.example.com/login_page', false ],
+ [ 'https://test.example.com/page/*', 'https://test.example.com/page/login_page', true ],
+ [ 'https://test.example.com/page/another_page/*', 'https://test.example.com/page/login', false ],
+ [ 'https://test.example.com/path/another/a/', 'https://test.example.com/path/another/a/', true ],
+ [ 'https://test.example.com/path/another/a/', 'https://test.example.com/path/another/b/', false ],
+ ];
+
+ for (const m of matches) {
+ assertRegex(siteMatch(m[0], m[1]), m[2], testCard, `siteMatch() for ${m[1]}`);
+ }
+}
+
+// Input field matching (keepassxc-browser.js)
+async function testInputFields() {
+ // Local filename, expected fields, action element ID (a button to be clicked)
+ const localFiles = [
+ [ 'html/basic1.html', 2 ], // Username/passwd fields
+ [ 'html/basic2.html', 1 ], // Only username field
+ [ 'html/basic3.html', 1 ], // Only password field
+ [ 'html/basic4.html', 3 ], // Username/passwd/TOTP fields
+ [ 'html/div1.html', 2, '#toggle' ], // Fields are behind a button that must be pressed
+ [ 'html/div2.html', 2, '#toggle' ], // Fields are behind a button that must be pressed behind a JavaScript
+ [ 'html/div3.html', 2, '#toggle' ], // Fields are behind a button that must be pressed
+ [ 'html/div4.html', 2, '#toggle' ], // Fields are behind a button that must be pressed
+ [ 'html/hidden_fields1.html', 0 ], // Two hidden fields
+ [ 'html/hidden_fields2.html', 1 ], // Two hidden fields with one visible
+ ];
+
+ for (const file of localFiles) {
+ await assertInputFields(file[0], file[1], file[2]);
+ }
+
+ document.getElementById('testFile').hidden = true;
+}
+
+// Search fields (kpxcFields
+async function testSearchFields() {
+ const searchFields = [
+ [ '', { id: 'otp_field', name: 'otp', type: 'text', maxLength: '8' }, 'Generic 2FA field', false ],
+ [ '', { placeholder: 'search', type: 'text', id: 'username' }, 'Placeholder only', true ],
+ [ '', { ariaLabel: 'search', type: 'text', id: 'username' }, 'aria-label only', true ],
+
+ ];
+
+ for (const field of searchFields) {
+ assertSearchField(field[0], field[1], field[2], field[3]);
+ }
+
+ assertSearchForm({ id: 'username', type: 'text', }, 'Generic input field under search form', true);
+}
+
+// TOTP fields (kpxcTOTPIcons)
+async function testTotpFields() {
+ const totpFields = [
+ [ '', { id: 'otp_field', name: 'otp', type: 'text', maxLength: '8' }, 'Generic 2FA field', true ],
+ [ '', { id: '2fa', type: 'text', maxLength: '6' }, 'Generic 2FA field', true ],
+ [ '', { id: '2fa', type: 'text', maxLength: '4' }, 'Ignore if field maxLength too small', false ],
+ [ '', { id: '2fa', type: 'text', maxLength: '12' }, 'Ignore if field maxLength too long', false ],
+ [ '', { id: 'username', type: 'text', }, 'Ignore a generic input field', false ],
+ [ '', { type: 'password', }, 'Ignore a password input field', false ],
+ [ // Protonmail
+ 'TwoFA-input ng-empty ng-invalid ng-invalid-required ng-valid-minlength ng-valid-maxlength ng-touched',
+ { autocapitalize: 'off', autocorrect: 'off', id: 'twoFactorCode', type: 'text', placeholder: 'Two-factor passcode', name: 'twoFactorCode' },
+ 'Protonmail 2FA',
+ true
+ ],
+ [ // Nextcloud
+ '',
+ { minlength: '6', maxLength: '10', name: 'challenge', placeholder: 'Authentication code', type: 'tel', },
+ 'Nextcloud 2FA',
+ true
+ ],
+ [ // GMail
+ 'whsOnd zHQkBf',
+ { autocomplete: 'off', id: 'idvPin', tabindex: '0', name: 'idvPin', pattern: '[0-9 ]*', type: 'tel', spellcheck: 'false' },
+ 'GMail 2FA',
+ true
+ ],
+ [ // Live.com
+ 'form-control',
+ { autocomplete: 'off', id: 'idTxtBx_SAOTCC_OTC', maxLength: '8', tabindex: '0', name: 'otc', placeholder: 'Code', type: 'tel' },
+ 'Live.com 2FA',
+ true
+ ],
+ ];
+
+ for (const field of totpFields) {
+ assertTOTPField(field[0], field[1], field[2], field[3]);
+ }
+}
+
+// Password change
+async function testPasswordChange() {
+ // Local filename, expected new password
+ const localFiles = [
+ [ 'html/passwordchange1.html', 'newPassword' ], // Default order without form
+ [ 'html/passwordchange2.html', 'newPassword' ], // Reversed order without form
+ [ 'html/passwordchange3.html', 'newPassword' ], // Default order with form
+ [ 'html/passwordchange4.html', 'newPassword' ], // Reversed order with form
+ [ 'html/passwordchange5.html', 'newPassword' ], // Each field has own form
+ ];
+
+ for (const file of localFiles) {
+ await assertPasswordChangeFields(file[0], file[1]);
+ }
+
+ document.getElementById('testFile').hidden = true;
+}
+
+// Run tests
+(async () => {
+ await Promise.all([
+ await testGeneral(),
+ await testInputFields(),
+ await testSearchFields(),
+ await testTotpFields(),
+ await testPasswordChange(),
+ ]);
+})();