mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2024-11-19 18:11:36 +01:00
224 lines
8.0 KiB
C#
224 lines
8.0 KiB
C#
using System;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Models;
|
|
using BTCPayServer.Models.ManageViewModels;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace BTCPayServer.Controllers
|
|
{
|
|
public partial class ManageController
|
|
{
|
|
private const string RecoveryCodesKey = nameof(RecoveryCodesKey);
|
|
private const string AuthenicatorUriFormat = "otpauth://totp/{0}:{1}?secret={2}&issuer={0}&digits=6";
|
|
|
|
[HttpGet]
|
|
public async Task<IActionResult> TwoFactorAuthentication()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
var model = new TwoFactorAuthenticationViewModel
|
|
{
|
|
HasAuthenticator = await _userManager.GetAuthenticatorKeyAsync(user) != null,
|
|
Is2faEnabled = user.TwoFactorEnabled,
|
|
RecoveryCodesLeft = await _userManager.CountRecoveryCodesAsync(user),
|
|
};
|
|
|
|
return View(model);
|
|
}
|
|
|
|
[HttpGet]
|
|
public async Task<IActionResult> Disable2faWarning()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
if (!user.TwoFactorEnabled)
|
|
{
|
|
throw new ApplicationException(
|
|
$"Unexpected error occurred disabling 2FA for user with ID '{user.Id}'.");
|
|
}
|
|
|
|
return View(nameof(Disable2fa));
|
|
}
|
|
|
|
[HttpPost]
|
|
[ValidateAntiForgeryToken]
|
|
public async Task<IActionResult> Disable2fa()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
var disable2faResult = await _userManager.SetTwoFactorEnabledAsync(user, false);
|
|
if (!disable2faResult.Succeeded)
|
|
{
|
|
throw new ApplicationException(
|
|
$"Unexpected error occurred disabling 2FA for user with ID '{user.Id}'.");
|
|
}
|
|
|
|
_logger.LogInformation("User with ID {UserId} has disabled 2fa.", user.Id);
|
|
return RedirectToAction(nameof(TwoFactorAuthentication));
|
|
}
|
|
|
|
[HttpGet]
|
|
public async Task<IActionResult> EnableAuthenticator()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
var model = new EnableAuthenticatorViewModel();
|
|
await LoadSharedKeyAndQrCodeUriAsync(user, model);
|
|
|
|
return View(model);
|
|
}
|
|
|
|
[HttpPost]
|
|
[ValidateAntiForgeryToken]
|
|
public async Task<IActionResult> EnableAuthenticator(EnableAuthenticatorViewModel model)
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
{
|
|
await LoadSharedKeyAndQrCodeUriAsync(user, model);
|
|
return View(model);
|
|
}
|
|
|
|
// Strip spaces and hypens
|
|
var verificationCode = model.Code.Replace(" ", string.Empty, StringComparison.OrdinalIgnoreCase).Replace("-", string.Empty, StringComparison.OrdinalIgnoreCase);
|
|
|
|
var is2faTokenValid = await _userManager.VerifyTwoFactorTokenAsync(
|
|
user, _userManager.Options.Tokens.AuthenticatorTokenProvider, verificationCode);
|
|
|
|
if (!is2faTokenValid)
|
|
{
|
|
ModelState.AddModelError("Code", "Verification code is invalid.");
|
|
await LoadSharedKeyAndQrCodeUriAsync(user, model);
|
|
return View(model);
|
|
}
|
|
|
|
await _userManager.SetTwoFactorEnabledAsync(user, true);
|
|
_logger.LogInformation("User with ID {UserId} has enabled 2FA with an authenticator app.", user.Id);
|
|
var recoveryCodes = await _userManager.GenerateNewTwoFactorRecoveryCodesAsync(user, 10);
|
|
TempData[RecoveryCodesKey] = recoveryCodes.ToArray();
|
|
|
|
return RedirectToAction(nameof(GenerateRecoveryCodes));
|
|
}
|
|
|
|
[HttpGet]
|
|
public IActionResult ResetAuthenticatorWarning()
|
|
{
|
|
return View(nameof(ResetAuthenticator));
|
|
}
|
|
|
|
[HttpPost]
|
|
[ValidateAntiForgeryToken]
|
|
public async Task<IActionResult> ResetAuthenticator()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
await _userManager.SetTwoFactorEnabledAsync(user, false);
|
|
await _userManager.ResetAuthenticatorKeyAsync(user);
|
|
_logger.LogInformation("User with id '{UserId}' has reset their authentication app key.", user.Id);
|
|
|
|
return RedirectToAction(nameof(EnableAuthenticator));
|
|
}
|
|
|
|
[HttpGet]
|
|
public IActionResult GenerateRecoveryCodes()
|
|
{
|
|
var recoveryCodes = (string[])TempData[RecoveryCodesKey];
|
|
if (recoveryCodes == null)
|
|
{
|
|
return RedirectToAction(nameof(TwoFactorAuthentication));
|
|
}
|
|
|
|
var model = new GenerateRecoveryCodesViewModel {RecoveryCodes = recoveryCodes};
|
|
return View(model);
|
|
}
|
|
|
|
[HttpGet]
|
|
public async Task<IActionResult> GenerateRecoveryCodesWarning()
|
|
{
|
|
var user = await _userManager.GetUserAsync(User);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{_userManager.GetUserId(User)}'.");
|
|
}
|
|
|
|
if (!user.TwoFactorEnabled)
|
|
{
|
|
throw new ApplicationException($"Cannot generate recovery codes for user with ID '{user.Id}' because they do not have 2FA enabled.");
|
|
}
|
|
|
|
return View(nameof(GenerateRecoveryCodesWarning));
|
|
}
|
|
|
|
private string GenerateQrCodeUri(string email, string unformattedKey)
|
|
{
|
|
return string.Format(CultureInfo.InvariantCulture,
|
|
AuthenicatorUriFormat,
|
|
_urlEncoder.Encode("BTCPayServer"),
|
|
_urlEncoder.Encode(email),
|
|
unformattedKey);
|
|
}
|
|
|
|
private string FormatKey(string unformattedKey)
|
|
{
|
|
var result = new StringBuilder();
|
|
int currentPosition = 0;
|
|
while (currentPosition + 4 < unformattedKey.Length)
|
|
{
|
|
result.Append(unformattedKey.Substring(currentPosition, 4)).Append(" ");
|
|
currentPosition += 4;
|
|
}
|
|
|
|
if (currentPosition < unformattedKey.Length)
|
|
{
|
|
result.Append(unformattedKey.Substring(currentPosition));
|
|
}
|
|
|
|
return result.ToString().ToLowerInvariant();
|
|
}
|
|
|
|
|
|
private async Task LoadSharedKeyAndQrCodeUriAsync(ApplicationUser user, EnableAuthenticatorViewModel model)
|
|
{
|
|
var unformattedKey = await _userManager.GetAuthenticatorKeyAsync(user);
|
|
if (string.IsNullOrEmpty(unformattedKey))
|
|
{
|
|
await _userManager.ResetAuthenticatorKeyAsync(user);
|
|
unformattedKey = await _userManager.GetAuthenticatorKeyAsync(user);
|
|
}
|
|
|
|
model.SharedKey = FormatKey(unformattedKey);
|
|
model.AuthenticatorUri = GenerateQrCodeUri(user.Email, unformattedKey);
|
|
}
|
|
}
|
|
}
|