diff --git a/modules/2fa/modules.php b/modules/2fa/modules.php index 67dca88b22..18354a13ce 100644 --- a/modules/2fa/modules.php +++ b/modules/2fa/modules.php @@ -109,6 +109,33 @@ public function process() { } } +/** + * Verify 2fa code is paired with Authenticator app before enabling 2fa + * @subpackage 2fa/handler + */ +class Hm_Handler_2fa_setup_check extends Hm_Handler_Module { + public function process() { + + list($secret, $simple) = get_2fa_key($this->config); + if (!$secret) { + Hm_Debug::add('2FA module set enabled, but no shared secret configured'); + return; + } + + $verified = false; + $len = $simple ? 15 : 64; + + $username = $this->session->get('username', false); + $secret = create_secret($secret, $username, $len); + + if (check_2fa_pin($this->request->post['2fa_code'], $secret)) { + $verified = true; + } + + $this->out('ajax_2fa_verified', $verified); + } +} + /** * @subpackage 2fa/output */ @@ -120,40 +147,63 @@ protected function output() { if (array_key_exists('2fa_enable', $settings)) { $enabled = $settings['2fa_enable']; } - $res = ''. - ''.$this->trans('2 Factor Authentication').''; + $res = ' + '. + ''.$this->trans('2 Factor Authentication').' + + '; + + $res .= ' + + '.$this->trans('Enable 2 factor authentication').' + '; - $res .= ''.$this->trans('Enable 2 factor authentication'). - 'get('2fa_svg'); + if ($svg) { - $qr_code = ''; + $qr_code = ''; if (!$enabled) { - $qr_code .= '
'. - $this->trans('Configure your authentication app using the barcode below BEFORE enabling 2 factor authentication.').'
'; + $qr_code .= '
'.$this->trans('Configure your authentication app using the barcode below BEFORE enabling 2 factor authentication.').'
'; } else { $qr_code .= '
'.$this->trans('Update your settings with the code below').'
'; } $qr_code .= $svg; - $qr_code .= ''; - $qr_code .= ''.$this->trans('If you can\'t use the QR code, you can enter the code below manually (no line breaks)').''; - $qr_code .= ''.wordwrap($this->html_safe($this->get('2fa_secret', '')), 60, '
', true).''; + $qr_code .= '
'.$this->trans('If you can\'t use the QR code, you can enter the code below manually (no line breaks)').'
'; + $qr_code .= wordwrap($this->html_safe($this->get('2fa_secret', '')), 60, '
', true); } else { - $qr_code = ''.$this->trans('Unable to generate 2 factor authentication QR code').''; + $qr_code = '
'.$this->trans('Unable to generate 2 factor authentication QR code').'
'; } - $res .= $qr_code; - $res .= ''.$this->trans('The following backup codes can be used to access your account if you lose your device').'

'; + $res .= $qr_code . ''; + + $res .= '
'.$this->trans('The following backup codes can be used to access your account if you lose your device'). '
'; + foreach ($backup_codes as $val) { $res .= ' '.$val.'
'; } - $res .= ''; + $res .= '
+
+ Enter the confirmation code +
+
+
+ + + + + + +
+ +
+
'.$this->trans('Enter the 6 digit code from your Authenticator application').'
+
+
+
+ + '; return $res; } } diff --git a/modules/2fa/setup.php b/modules/2fa/setup.php index df0749de74..e3c9920b66 100644 --- a/modules/2fa/setup.php +++ b/modules/2fa/setup.php @@ -14,8 +14,20 @@ add_handler('settings', 'process_enable_2fa', true, '2fa', 'save_user_settings', 'before'); add_output('settings', 'enable_2fa_setting', true, '2fa', 'end_settings_form', 'before'); -return array( 'allowed_post' => array( - '2fa_code' => FILTER_SANITIZE_FULL_SPECIAL_CHARS, - '2fa_enable' => FILTER_VALIDATE_INT, - '2fa_backup_codes' => array('filter' => FILTER_VALIDATE_INT, 'flags' => FILTER_FORCE_ARRAY) -)); +/* 2fa setup handler */ +setup_base_ajax_page('ajax_2fa_setup_check', 'core'); +add_handler('ajax_2fa_setup_check', '2fa_setup_check', true); + +return array( + 'allowed_pages' => array( + 'ajax_2fa_setup_check', + ), + 'allowed_post' => array( + '2fa_code' => FILTER_SANITIZE_FULL_SPECIAL_CHARS, + '2fa_enable' => FILTER_VALIDATE_INT, + '2fa_backup_codes' => array('filter' => FILTER_VALIDATE_INT, 'flags' => FILTER_FORCE_ARRAY) + ), + 'allowed_output' => array( + 'ajax_2fa_verified' => array(FILTER_VALIDATE_BOOLEAN, false), + ), +); diff --git a/modules/2fa/site.css b/modules/2fa/site.css index 73601abafe..e7a856aa2b 100644 --- a/modules/2fa/site.css +++ b/modules/2fa/site.css @@ -1,4 +1,219 @@ -.tfa_error, .tfa_input { margin-left: 20px; } -.tfa_error { color: red; margin-right: 20px; margin-bottom: 10px; } -.tfa_setting { display: none; } -.mobile .tfa_input { max-width: 340px; margin-bottom: 10px; } +:root { + --tfa-form-valid-color: #198754; + --tfa-form-valid-border-color: #198754; + --tfa-form-invalid-color: #dc3545; + --tfa-form-invalid-border-color: #dc3545; + --tfa-border-radius: 0.375rem; + --tfa-border-color: #dee2e6; + --tfa-base-color: #666; + --tfa-secondary-color: rgba(33, 37, 41, 0.75); +} + +.tfa_error, +.tfa_input { + margin-left: 20px; +} + +.tfa_error { + color: red; + margin-right: 20px; + margin-bottom: 10px; +} + +.tfa_setting { + display: none; +} + +.user_settings .tfa_setting td { + height: auto; +} + +.mobile .tfa_input { + max-width: 340px; + margin-bottom: 10px; +} + +.tfa_mt_1 { + margin-top: 0.7rem; +} + +.tfa_mb_1 { + margin-bottom: 0.7rem; +} + +.tfa_confirmation_fieldset { + color: var(--tfa-base-color); + border-radius: var(--tfa-border-radius); + border: 1px solid var(--tfa-border-color); + max-width: 365px; +} + +.tfa_confirmation_wrapper .tfa_confirmation_hint { + font-style: italic; + font-size: 0.75em; + margin-top: 0.8rem; + color: var(--tfa-secondary-color); +} +.tfa_confirmation_wrapper .tfa_confirmation_hint .invalid { + color: var(--tfa-form-invalid-color); +} +.tfa_confirmation_wrapper .valid { + border-color: var(--tfa-form-valid-color); + padding-right: calc(1.5em + 0.75rem); + background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23198754' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e"); + background-repeat: no-repeat; + background-position: right calc(0.375em + 0.1875rem) center; + background-size: calc(0.75em + 0.375rem) calc(0.75em + 0.375rem); +} +.tfa_confirmation_wrapper .invalid { + border-color: var(--tfa-form-invalid-color); +} +.tfa_confirmation_wrapper input::-webkit-outer-spin-button, +.tfa_confirmation_wrapper input::-webkit-inner-spin-button { + -webkit-appearance: none; + margin: 0; +} +.tfa_confirmation_wrapper input[type=number] { + -moz-appearance: textfield; +} +.tfa_confirmation_wrapper input, +.tfa_confirmation_wrapper button, +.tfa_confirmation_wrapper a { + -webkit-tap-highlight-color: rgba(0, 0, 0, 0); +} + +.tfa_confirmation_input_digits { + --total-digits: 6; + display: grid; + grid-template-columns: repeat(var(--total-digits), 1fr); + gap: 0.5rem; +} +.tfa_confirmation_input_digit { + display: block; + width: 2rem; + height: 2rem; + padding: 0.625rem; + border: 1px solid var(--tfa-border-color); + border-radius: var(--tfa-border-radius); + font-size: 1.5rem; + text-align: center; +} +.tfa_confirmation_input_digit:focus { + color: #212529; + background-color: #fff; + border-color: #86b7fe; + outline: 0; + box-shadow: 0 0 0 0.25rem rgba(13, 110, 253, 0.25); +} +.tfa_confirmation_input_button { + padding: 10px; + color: var(--tfa-base-color); + border-radius: var(--tfa-border-radius); + border: 1px solid var(--tfa-border-color); + background-color: #fff; + text-decoration: none; + margin-top: 0.8rem; + font-size: 0.9rem; +} +.tfa_confirmation_input_button:hover { + cursor: pointer; +} +.tfa_confirmation_input_button.invalid { + padding-right: calc(1.5em + 0.75rem); + background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23dc3545'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e"); + background-repeat: no-repeat; + background-position: right calc(0.375em + 0.1875rem) center; + background-size: calc(0.75em + 0.375rem) calc(0.75em + 0.375rem); + animation: shake 150ms 2 linear; + -moz-animation: shake 150ms 2 linear; + -webkit-animation: shake 150ms 2 linear; + -o-animation: shake 150ms 2 linear; +} +.tfa_confirmation_input_button.loading { + cursor: not-allowed; + opacity: 0.7; + padding-right: calc(1.75em + 0.75rem); + background-image: url(""); + background-repeat: no-repeat; + background-position: right calc(0.375em + 0.1875rem) center; + background-size: calc(1.5em + 0.375rem) calc(1.5em + 0.375rem); +} + +@media (max-width: 767px) { + .hidden-xs { + display: none !important; + } + + .tfa_confirmation_fieldset { + max-width: 100vh; + } + .tfa_confirmation_fieldset .tfa_confirmation_input_digit { + width: 1.5rem; + height: 1.5rem; + } + + .user_settings .tfa_setting td { + display: block; + } +} +/** +** Shake Annimation Starts Here. +**/ +@keyframes shake { + 0% { + transform: translate(3px, 0); + } + 50% { + transform: translate(-3px, 0); + } + 100% { + transform: translate(0, 0); + } +} +@-moz-keyframes shake { + 0% { + -moz-transform: translate(3px, 0); + } + 50% { + -moz-transform: translate(-3px, 0); + } + 100% { + -moz-transform: translate(0, 0); + } +} +@-webkit-keyframes shake { + 0% { + -webkit-transform: translate(3px, 0); + } + 50% { + -webkit-transform: translate(-3px, 0); + } + 100% { + -webkit-transform: translate(0, 0); + } +} +@-ms-keyframes shake { + 0% { + -ms-transform: translate(3px, 0); + } + 50% { + -ms-transform: translate(-3px, 0); + } + 100% { + -ms-transform: translate(0, 0); + } +} +@-o-keyframes shake { + 0% { + -o-transform: translate(3px, 0); + } + 50% { + -o-transform: translate(-3px, 0); + } + 100% { + -o-transform: translate(0, 0); + } +} +/** + ** Shake Annimation Ends Here. + **/ \ No newline at end of file diff --git a/modules/2fa/site.js b/modules/2fa/site.js new file mode 100644 index 0000000000..ec8b68ec70 --- /dev/null +++ b/modules/2fa/site.js @@ -0,0 +1,175 @@ +$(function () { + function tFaToast(message, timer = 3000) { + Hm_Notices.show([message]); + const tm = setTimeout(function () { + Hm_Notices.hide(true); + clearTimeout(tm); + }, timer); + } + + function focusElement(elem) { + elem.focus(); + elem.select(); + } + + function validateInput(input) { + if (isNaN(Number(input.value)) || input.value === "") { + input.classList.add("invalid"); + return false; + } + input.classList.remove("invalid"); + return input.value.length > 1 ? input.value[0] : input.value; + } + + function focusNextInput(currentInput) { + if (currentInput.nextElementSibling) { + focusElement(currentInput.nextElementSibling); + } + } + + function focusPrevInput(currentInput) { + if (currentInput.previousElementSibling) { + focusElement(currentInput.previousElementSibling); + } + } + + function handleArrowKeys(e) { + if (e.key == "ArrowRight") { + e.preventDefault(); + focusNextInput(e.target); + } + if (e.key == "ArrowLeft") { + e.preventDefault(); + focusPrevInput(e.target); + } + } + + function handleBackspace(e) { + if (e.key == "Backspace") { + e.target.value = ""; + e.target.classList.remove("invalid"); + focusPrevInput(e.target); + } + } + + function getInputCode() { + let inputCode = ""; + for (let input of $(".tfa_confirmation_input_digit")) { + input.classList.remove("invalid"); + if (validateInput(input) === false) { + return null; + } + inputCode += input.value; + input.blur(); + } + return inputCode; + } + + function tFaInit() { + let verified = false; + + const formInput = $('input[name="2fa_enable"]'); + const tfaEnabled = formInput.is(":checked"); + const confirmationBtn = $("#tfaConfirmationBtn"); + const formContainer = $(".tfa_confirmation_form"); + + if (!tfaEnabled) { + formInput.on("change", function () { + const checked = $(this).is(":checked"); + + if (checked && !verified) { + $(this).prop("checked", false); + tFaToast("ERRYou need to verify your 2 factor authentication code before processing"); + return; + } + }); + } + + formContainer.on("input", function (e) { + const value = validateInput(e.target); + if (value !== false) { + e.target.value = value; + focusNextInput(e.target); + } else { + e.target.value = ""; + } + }); + + formContainer.on("click", function (e) { + if (e.target.tagName == "INPUT") { + focusElement(e.target); + } + }); + + formContainer.on("keydown", ".tfa_confirmation_input_digit", function (e) { + handleArrowKeys(e); + handleBackspace(e); + }); + + formContainer.on("paste", function (e) { + const pastedData = e.originalEvent.clipboardData.getData("text"); + + if (pastedData === "") { + return; + } + const payloadData = pastedData.split(""); + + const inputs = $(".tfa_confirmation_input_digit"); + + inputs.each(function (i, input) { + $(input).val(payloadData[i]).focus(); + }); + + const tm = setTimeout(() => { + for (let input of inputs) { + if (validateInput(input) === false) { + focusElement(input); + break; + } + } + clearTimeout(tm); + }, 0); + }); + + confirmationBtn.on("click", function (e) { + e.preventDefault(); + $(this).removeClass("invalid").removeClass("shake"); + $(".tfa_confirmation_input_digit").removeClass("invalid"); + var code = getInputCode(); + + if (!code) { + tFaToast("ERRYou need to enter the verification code"); + var tm = setTimeout(function () { + Hm_Notices.hide(true); + clearTimeout(tm); + }, 2000); + return; + } + + $(this).text("Processing").addClass("loading"); + Hm_Ajax.request( + [ + { name: "hm_ajax_hook", value: "ajax_2fa_setup_check" }, + { name: "2fa_code", value: code }, + ], + function (response) { + if (response && response.ajax_2fa_verified) { + verified = true; + formInput.prop("checked", true); + confirmationBtn.addClass("valid"); + tFaToast("2 factor authentication enabled"); + } else { + verified = false; + formInput.prop("checked", false); + $(".tfa_confirmation_input_digit").addClass("invalid"); + confirmationBtn.addClass("invalid").addClass("shake"); + tFaToast("ERR2 factor authentication code does not match"); + } + confirmationBtn.text("Verify code").removeClass("loading"); + } + ); + }); + } + + tFaInit(); +});