mirror of
https://github.com/keepassxreboot/keepassxc-browser.git
synced 2026-03-11 08:54:43 +00:00
121 lines
4.8 KiB
JavaScript
121 lines
4.8 KiB
JavaScript
'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', { id: text });
|
|
const span = kpxcUI.createElement('span', '', '', text);
|
|
const br = document.createElement('br');
|
|
|
|
document.querySelector(card).appendMultiple(icon, span, br);
|
|
}
|
|
|
|
// Input field matching (keepassxc-browser.js)
|
|
async function testInputFields() {
|
|
// Div ID, expected fields, action element ID (a button to be clicked)
|
|
const testDivs = [
|
|
[ 'basic1', 2 ], // Username/passwd fields
|
|
[ 'basic2', 1 ], // Only username field
|
|
[ 'basic3', 1 ], // Only password field
|
|
[ 'basic4', 3 ], // Username/passwd/TOTP fields
|
|
[ 'div1', 2, '#toggle1' ], // Fields are behind a button that must be pressed
|
|
[ 'div2', 2, '#toggle2' ], // Fields are behind a button that must be pressed behind a JavaScript
|
|
[ 'div3', 2, '#toggle3' ], // Fields are behind a button that must be pressed
|
|
[ 'div4', 2, '#toggle4' ], // Fields are behind a button that must be pressed
|
|
[ 'hiddenFields1', 0 ], // Two hidden fields
|
|
[ 'hiddenFields2', 1 ], // Two hidden fields with one visible
|
|
];
|
|
|
|
for (const div of testDivs) {
|
|
await assertInputFields(div[0], div[1], div[2]);
|
|
}
|
|
}
|
|
|
|
// 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: '2fa', type: 'text', maxLength: '12', autocomplete: 'one-time-code' }, 'Accept if one-time-code', true ],
|
|
[ '', { 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() {
|
|
// Div ID, expected new password
|
|
const localDivs = [
|
|
[ 'passwordChange1', 'newPassword' ], // Default order without form
|
|
[ 'passwordChange2', 'newPassword' ], // Reversed order without form
|
|
[ 'passwordChange3', 'newPassword' ], // Default order with form
|
|
[ 'passwordChange4', 'newPassword' ], // Reversed order with form
|
|
[ 'passwordChange5', 'newPassword' ], // Each field has own form
|
|
];
|
|
|
|
for (const div of localDivs) {
|
|
await assertPasswordChangeFields(div[0], div[1]);
|
|
}
|
|
}
|
|
|
|
// Run tests
|
|
(async () => {
|
|
await Promise.all([
|
|
await testInputFields(),
|
|
await testSearchFields(),
|
|
await testTotpFields(),
|
|
await testPasswordChange(),
|
|
]);
|
|
})();
|