ui: rework TFA prompt on login

Improve UX by avoiding the need to click some buttons twice, or
calling TOTP and Recovery codes both "OTP" codes and showing multiple
buttons, with all having the same goal "submit a TFA token" at the
same time.

Instead use a tab panel with a single submit button.

WebAuthn can and should be still improved, but that can be OK as
followup.

Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
This commit is contained in:
Thomas Lamprecht 2021-01-27 13:19:09 +01:00
parent 651a61f559
commit f91481eded

View File

@ -241,61 +241,95 @@ Ext.define('PBS.login.TfaWindow', {
extend: 'Ext.window.Window', extend: 'Ext.window.Window',
mixins: ['Proxmox.Mixin.CBind'], mixins: ['Proxmox.Mixin.CBind'],
modal: true,
resizable: false,
title: gettext("Second login factor required"), title: gettext("Second login factor required"),
cancelled: true, modal: true,
resizable: false,
width: 512, width: 512,
layout: { layout: {
type: 'vbox', type: 'vbox',
align: 'stretch', align: 'stretch',
}, },
defaultButton: 'totpButton', defaultButton: 'tfaButton',
viewModel: {
data: {
canConfirm: false,
availabelChallenge: {},
},
},
cancelled: true,
controller: { controller: {
xclass: 'Ext.app.ViewController', xclass: 'Ext.app.ViewController',
init: function(view) { init: function(view) {
let me = this; let me = this;
let vm = me.getViewModel();
if (!view.userid) { if (!view.userid) {
throw "no userid given"; throw "no userid given";
} }
if (!view.ticket) { if (!view.ticket) {
throw "no ticket given"; throw "no ticket given";
} }
const challenge = view.challenge;
if (!view.challenge) { if (!challenge) {
throw "no challenge given"; throw "no challenge given";
} }
if (!view.challenge.webauthn) { let firstAvailableTab = -1, i = 0;
me.lookup('webauthnButton').setVisible(false); for (const k of ['webauthn', 'totp', 'recovery']) {
const available = !!challenge[k];
vm.set(`availabelChallenge.${k}`, available);
if (firstAvailableTab < 0 && available) {
firstAvailableTab = i;
}
i++;
}
view.down('tabpanel').setActiveTab(firstAvailableTab);
if (challenge.recovery) {
me.lookup('availableRecovery').update(Ext.String.htmlEncode(
gettext('Available recovery keys: ') + view.challenge.recovery.join(', '),
));
me.lookup('availableRecovery').setVisible(true);
if (view.challenge.recovery.length <= 3) {
me.lookup('recoveryLow').setVisible(true);
}
} }
if (!view.challenge.totp) { if (challenge.webauthn) {
me.lookup('totpButton').setVisible(false);
}
if (!view.challenge.recovery || !view.challenge.recovery.length) {
me.lookup('recoveryButton').setVisible(false);
} else if (view.challenge.recovery.length <= 3) {
me.lookup('recoveryButton')
.setIconCls('fa fa-fw fa-exclamation-triangle');
}
if (!view.challenge.totp && !view.challenge.recovery) {
// only webauthn tokens available, maybe skip ahead?
me.lookup('totp').setVisible(false);
me.lookup('waiting').setVisible(true);
let _promise = me.loginWebauthn(); let _promise = me.loginWebauthn();
} }
}, },
control: {
'tabpanel': {
tabchange: function(tabPanel, newCard, oldCard) {
// for now every TFA method has at max one field, so keep it simple..
let oldField = oldCard.down('field');
if (oldField) {
oldField.setDisabled(true);
}
let newField = newCard.down('field');
if (newField) {
newField.setDisabled(false);
newField.focus();
newField.validate();
}
},
},
'field': {
validitychange: function(field, valid) {
// triggers only for enabled fields and we disable the one from the
// non-visible tab, so we can just directly use the valid param
this.getViewModel().set('canConfirm', valid);
},
},
},
onClose: function() { onClose: function() {
let me = this; let me = this;
@ -315,24 +349,15 @@ Ext.define('PBS.login.TfaWindow', {
loginTotp: function() { loginTotp: function() {
let me = this; let me = this;
let _promise = me.finishChallenge('totp:' + me.lookup('totp').value); let code = me.lookup('totp').getValue();
let _promise = me.finishChallenge(`totp:${code}`);
}, },
loginWebauthn: async function() { loginWebauthn: async function() {
let me = this; let me = this;
let view = me.getView(); let view = me.getView();
// avoid this window ending up above the tfa popup if we got triggered from init(). me.lookup('webAuthnWaiting').setVisible(true);
await PBS.Async.sleep(100);
// FIXME: With webauthn the browser provides a popup (since it doesn't necessarily need
// to require pressing a button, but eg. use a fingerprint scanner or face detection
// etc., so should we just trust that that happens and skip the popup?)
let msg = Ext.Msg.show({
title: `Webauthn: ${gettext('Login')}`,
message: gettext('Please press the button on your Authenticator Device'),
buttons: [],
});
let challenge = view.challenge.webauthn; let challenge = view.challenge.webauthn;
@ -343,14 +368,21 @@ Ext.define('PBS.login.TfaWindow', {
cred.id = PBS.Utils.base64url_to_bytes(cred.id); cred.id = PBS.Utils.base64url_to_bytes(cred.id);
} }
let controller = new AbortController();
challenge.signal = controller.signal;
let hwrsp; let hwrsp;
try { try {
//Promise.race( ...
hwrsp = await navigator.credentials.get(challenge); hwrsp = await navigator.credentials.get(challenge);
} catch (error) { } catch (error) {
view.onReject(error); view.onReject(error);
return; return;
} finally { } finally {
msg.close(); let waitingMessage = me.lookup('webAuthnWaiting');
if (waitingMessage) {
waitingMessage.setVisible(false);
}
} }
let response = { let response = {
@ -367,32 +399,21 @@ Ext.define('PBS.login.TfaWindow', {
}, },
}; };
msg.close();
await me.finishChallenge("webauthn:" + JSON.stringify(response)); await me.finishChallenge("webauthn:" + JSON.stringify(response));
}, },
loginRecovery: function() { loginRecovery: function() {
let me = this; let me = this;
let view = me.getView();
if (me.login_recovery_confirm) { let key = me.lookup('recoveryKey').getValue();
let _promise = me.finishChallenge('recovery:' + me.lookup('totp').value); let _promise = me.finishChallenge(`recovery:${key}`);
} else { },
me.login_recovery_confirm = true;
me.lookup('totpButton').setVisible(false); loginTFA: function() {
me.lookup('webauthnButton').setVisible(false); let me = this;
me.lookup('recoveryButton').setText(gettext("Confirm")); let view = me.getView();
me.lookup('recoveryInfo').setVisible(true); let tfaPanel = view.down('tabpanel').getActiveTab();
console.log("RECOVERY:", view.challenge.recovery); me[tfaPanel.handler]();
me.lookup('availableRecovery').update(Ext.String.htmlEncode(
gettext('Available recovery keys: ') + view.challenge.recovery.join(', '),
));
me.lookup('availableRecovery').setVisible(true);
if (view.challenge.recovery.length <= 3) {
me.lookup('recoveryLow').setVisible(true);
}
}
}, },
finishChallenge: function(password) { finishChallenge: function(password) {
@ -424,81 +445,111 @@ Ext.define('PBS.login.TfaWindow', {
close: 'onClose', close: 'onClose',
}, },
items: [{
xtype: 'tabpanel',
region: 'center',
layout: 'fit',
bodyPadding: 10,
stateId: 'pbs-tfa-login-panel', // FIXME: do manually - get/setState miss
stateful: true,
stateEvents: ['tabchange'],
items: [ items: [
{ {
xtype: 'form', xtype: 'panel',
layout: 'anchor', title: 'WebAuthn',
border: false, iconCls: 'fa fa-fw fa-shield',
fieldDefaults: { handler: 'loginWebauthn',
anchor: '100%', bind: {
padding: '0 5', disabled: '{!availabelChallenge.webauthn}',
}, },
items: [ items: [
{ {
xtype: 'textfield', xtype: 'box',
fieldLabel: gettext('Please enter your OTP verification code:'), html: `<i class="fa fa-refresh fa-spin fa-fw"></i>` +
labelWidth: '300px', gettext('Please insert your authenticator device and press its button'),
name: 'totp',
reference: 'totp',
allowBlank: false,
},
],
}, },
{ {
xtype: 'box', xtype: 'box',
html: gettext('Waiting for second factor.'), html: gettext('Waiting for second factor.'),
reference: 'waiting', reference: 'webAuthnWaiting',
padding: '0 5',
hidden: true, hidden: true,
}, },
{ ],
xtype: 'box',
padding: '0 5',
reference: 'recoveryInfo',
hidden: true,
html: gettext('Please note that each recovery code can only be used once!'),
style: {
textAlign: "center",
},
}, },
{
xtype: 'panel',
title: gettext('TOTP App'),
iconCls: 'fa fa-fw fa-clock-o',
handler: 'loginTotp',
bind: {
disabled: '{!availabelChallenge.totp}',
},
items: [
{
xtype: 'textfield',
fieldLabel: gettext('Please enter your TOTP verification code'),
labelWidth: 300,
name: 'totp',
disabled: true,
reference: 'totp',
allowBlank: false,
regex: /^[0-9]{6}$/,
regexText: 'TOTP codes consist of six decimal digits',
},
],
},
{
xtype: 'panel',
title: gettext('Recovery Key'),
iconCls: 'fa fa-fw fa-file-text-o',
handler: 'loginRecovery',
bind: {
disabled: '{!availabelChallenge.recovery}',
},
items: [
{ {
xtype: 'box', xtype: 'box',
padding: '0 5',
reference: 'availableRecovery', reference: 'availableRecovery',
hidden: true, hidden: true,
style: {
textAlign: "center",
}, },
{
xtype: 'textfield',
fieldLabel: gettext('Please enter one of your single-use recovery keys'),
labelWidth: 300,
name: 'recoveryKey',
disabled: true,
reference: 'recoveryKey',
allowBlank: false,
regex: /^[0-9a-f]{4}(-[0-9a-f]{4}){3}$/,
regexText: 'Does not looks like a valid recovery key',
},
{
xtype: 'box',
reference: 'recoveryInfo',
hidden: true, // FIXME: remove this?
html: gettext('Note that each recovery code can only be used once!'),
}, },
{ {
xtype: 'box', xtype: 'box',
padding: '0 5',
reference: 'recoveryLow', reference: 'recoveryLow',
hidden: true, hidden: true,
html: '<i class="fa fa-exclamation-triangle warning"></i>' html: '<i class="fa fa-exclamation-triangle warning"></i>'
+ gettext('Only few recovery keys available. Please generate a new set!') + gettext('Less than {0} recovery keys available. Please generate a new set!'),
+ '<i class="fa fa-exclamation-triangle warning"></i>',
style: {
textAlign: "center",
},
}, },
], ],
},
],
}],
buttons: [ buttons: [
{ {
text: gettext('Login with TOTP'), text: gettext('Confirm Second Factor'),
handler: 'loginTotp', handler: 'loginTFA',
reference: 'totpButton', reference: 'tfaButton',
disabled: true,
bind: {
disabled: '{!canConfirm}',
}, },
{
text: gettext('Login with a recovery key'),
handler: 'loginRecovery',
reference: 'recoveryButton',
},
{
text: gettext('Use a Webauthn token'),
handler: 'loginWebauthn',
reference: 'webauthnButton',
}, },
], ],
}); });