2021-04-20 07:06:32 +02:00
|
|
|
using System;
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using BTCPayServer.Data;
|
2021-04-28 06:14:15 +02:00
|
|
|
using BTCPayServer.Fido2.Models;
|
2021-04-20 07:06:32 +02:00
|
|
|
using ExchangeSharp;
|
|
|
|
using Fido2NetLib;
|
|
|
|
using Fido2NetLib.Objects;
|
|
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
using NBitcoin;
|
2021-04-28 06:14:15 +02:00
|
|
|
using Newtonsoft.Json.Linq;
|
2024-12-03 08:47:26 +01:00
|
|
|
using static BTCPayServer.Fido2.Models.Fido2CredentialBlob;
|
2021-04-20 07:06:32 +02:00
|
|
|
|
|
|
|
namespace BTCPayServer.Fido2
|
|
|
|
{
|
|
|
|
public class Fido2Service
|
|
|
|
{
|
|
|
|
private static readonly ConcurrentDictionary<string, CredentialCreateOptions> CreationStore =
|
|
|
|
new ConcurrentDictionary<string, CredentialCreateOptions>();
|
|
|
|
private static readonly ConcurrentDictionary<string, AssertionOptions> LoginStore =
|
|
|
|
new ConcurrentDictionary<string, AssertionOptions>();
|
|
|
|
private readonly ApplicationDbContextFactory _contextFactory;
|
|
|
|
private readonly IFido2 _fido2;
|
2021-04-28 06:14:15 +02:00
|
|
|
private readonly Fido2Configuration _fido2Configuration;
|
2021-04-20 07:06:32 +02:00
|
|
|
|
2021-04-28 06:14:15 +02:00
|
|
|
public Fido2Service(ApplicationDbContextFactory contextFactory, IFido2 fido2, Fido2Configuration fido2Configuration)
|
2021-04-20 07:06:32 +02:00
|
|
|
{
|
|
|
|
_contextFactory = contextFactory;
|
|
|
|
_fido2 = fido2;
|
2021-04-28 06:14:15 +02:00
|
|
|
_fido2Configuration = fido2Configuration;
|
2021-04-20 07:06:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<CredentialCreateOptions> RequestCreation(string userId)
|
|
|
|
{
|
2021-12-31 08:59:02 +01:00
|
|
|
await using var dbContext = _contextFactory.CreateContext();
|
|
|
|
var user = await dbContext.Users.Include(applicationUser => applicationUser.Fido2Credentials)
|
|
|
|
.FirstOrDefaultAsync(applicationUser => applicationUser.Id == userId);
|
|
|
|
if (user == null)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. Get user existing keys by username
|
|
|
|
var existingKeys =
|
|
|
|
user.Fido2Credentials
|
|
|
|
.Where(credential => credential.Type == Fido2Credential.CredentialType.FIDO2)
|
2024-12-03 08:47:26 +01:00
|
|
|
.Select(c => c.GetFido2Blob().Descriptor?.ToFido2()).ToList();
|
2021-12-31 08:59:02 +01:00
|
|
|
|
|
|
|
// 3. Create options
|
|
|
|
var authenticatorSelection = new AuthenticatorSelection
|
|
|
|
{
|
|
|
|
RequireResidentKey = false,
|
|
|
|
UserVerification = UserVerificationRequirement.Preferred
|
|
|
|
};
|
2021-04-20 07:06:32 +02:00
|
|
|
|
2021-12-31 08:59:02 +01:00
|
|
|
var exts = new AuthenticationExtensionsClientInputs()
|
|
|
|
{
|
|
|
|
Extensions = true,
|
2024-12-03 08:47:26 +01:00
|
|
|
UserVerificationMethod = true
|
2021-12-31 08:59:02 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
var options = _fido2.RequestNewCredential(
|
|
|
|
new Fido2User() { DisplayName = user.UserName, Name = user.UserName, Id = user.Id.ToBytesUTF8() },
|
|
|
|
existingKeys, authenticatorSelection, AttestationConveyancePreference.None, exts);
|
|
|
|
|
|
|
|
// options.Rp = new PublicKeyCredentialRpEntity(Request.Host.Host, options.Rp.Name, "");
|
|
|
|
CreationStore.AddOrReplace(userId, options);
|
|
|
|
return options;
|
2021-04-20 07:06:32 +02:00
|
|
|
}
|
|
|
|
|
2021-04-28 06:14:15 +02:00
|
|
|
public async Task<bool> CompleteCreation(string userId, string name, string data)
|
2021-04-20 07:06:32 +02:00
|
|
|
{
|
2021-04-28 06:14:15 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
|
2024-12-03 08:47:26 +01:00
|
|
|
var attestationResponse = System.Text.Json.JsonSerializer.Deserialize<AuthenticatorAttestationRawResponse>(data);
|
2021-04-28 06:14:15 +02:00
|
|
|
await using var dbContext = _contextFactory.CreateContext();
|
|
|
|
var user = await dbContext.Users.Include(applicationUser => applicationUser.Fido2Credentials)
|
|
|
|
.FirstOrDefaultAsync(applicationUser => applicationUser.Id == userId);
|
|
|
|
if (user == null || !CreationStore.TryGetValue(userId, out var options))
|
|
|
|
{
|
2021-04-20 07:06:32 +02:00
|
|
|
return false;
|
2021-04-28 06:14:15 +02:00
|
|
|
}
|
2021-04-20 07:06:32 +02:00
|
|
|
|
|
|
|
// 2. Verify and make the credentials
|
2021-04-28 06:14:15 +02:00
|
|
|
var success =
|
2024-12-03 08:47:26 +01:00
|
|
|
await _fido2.MakeNewCredentialAsync(attestationResponse, options, (args, cancellation) => Task.FromResult(true));
|
2021-04-20 07:06:32 +02:00
|
|
|
|
|
|
|
// 3. Store the credentials in db
|
2021-12-31 08:59:02 +01:00
|
|
|
var newCredential = new Fido2Credential() { Name = name, ApplicationUserId = userId };
|
2021-04-28 06:14:15 +02:00
|
|
|
|
2021-04-20 07:06:32 +02:00
|
|
|
newCredential.SetBlob(new Fido2CredentialBlob()
|
|
|
|
{
|
2024-12-03 08:47:26 +01:00
|
|
|
Descriptor = new DescriptorClass(success.Result.CredentialId),
|
2021-04-20 07:06:32 +02:00
|
|
|
PublicKey = success.Result.PublicKey,
|
|
|
|
UserHandle = success.Result.User.Id,
|
|
|
|
SignatureCounter = success.Result.Counter,
|
|
|
|
CredType = success.Result.CredType,
|
|
|
|
AaGuid = success.Result.Aaguid.ToString(),
|
|
|
|
});
|
|
|
|
|
|
|
|
await dbContext.Fido2Credentials.AddAsync(newCredential);
|
|
|
|
await dbContext.SaveChangesAsync();
|
|
|
|
CreationStore.Remove(userId, out _);
|
|
|
|
return true;
|
|
|
|
|
2021-04-28 06:14:15 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
catch (Exception)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-20 07:06:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<List<Fido2Credential>> GetCredentials(string userId)
|
|
|
|
{
|
|
|
|
await using var context = _contextFactory.CreateContext();
|
|
|
|
return await context.Fido2Credentials
|
|
|
|
.Where(device => device.ApplicationUserId == userId)
|
|
|
|
.ToListAsync();
|
|
|
|
}
|
|
|
|
|
|
|
|
public async Task Remove(string id, string userId)
|
|
|
|
{
|
|
|
|
await using var context = _contextFactory.CreateContext();
|
2021-12-31 08:59:02 +01:00
|
|
|
var device = await context.Fido2Credentials.FindAsync(id);
|
2021-04-20 07:06:32 +02:00
|
|
|
if (device == null || !device.ApplicationUserId.Equals(userId, StringComparison.InvariantCulture))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
context.Fido2Credentials.Remove(device);
|
|
|
|
await context.SaveChangesAsync();
|
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<bool> HasCredentials(string userId)
|
|
|
|
{
|
|
|
|
await using var context = _contextFactory.CreateContext();
|
2021-11-11 13:03:08 +01:00
|
|
|
return await context.Fido2Credentials.Where(fDevice => fDevice.ApplicationUserId == userId && fDevice.Type == Fido2Credential.CredentialType.FIDO2).AnyAsync();
|
2021-04-20 07:06:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public async Task<AssertionOptions> RequestLogin(string userId)
|
|
|
|
{
|
|
|
|
await using var dbContext = _contextFactory.CreateContext();
|
|
|
|
var user = await dbContext.Users.Include(applicationUser => applicationUser.Fido2Credentials)
|
|
|
|
.FirstOrDefaultAsync(applicationUser => applicationUser.Id == userId);
|
|
|
|
if (!(user?.Fido2Credentials?.Any() is true))
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
var existingCredentials = user.Fido2Credentials
|
|
|
|
.Where(credential => credential.Type == Fido2Credential.CredentialType.FIDO2)
|
2024-12-03 08:47:26 +01:00
|
|
|
.Select(c => c.GetFido2Blob().Descriptor?.ToFido2())
|
2021-04-20 07:06:32 +02:00
|
|
|
.ToList();
|
|
|
|
var exts = new AuthenticationExtensionsClientInputs()
|
2021-12-31 08:59:02 +01:00
|
|
|
{
|
|
|
|
UserVerificationMethod = true,
|
2021-04-28 06:14:15 +02:00
|
|
|
Extensions = true,
|
2024-12-03 08:47:26 +01:00
|
|
|
AppID = _fido2Configuration.Origins.First()
|
2021-04-20 07:06:32 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// 3. Create options
|
|
|
|
var options = _fido2.GetAssertionOptions(
|
|
|
|
existingCredentials,
|
|
|
|
UserVerificationRequirement.Discouraged,
|
|
|
|
exts
|
|
|
|
);
|
|
|
|
LoginStore.AddOrReplace(userId, options);
|
|
|
|
return options;
|
|
|
|
}
|
2021-12-31 08:59:02 +01:00
|
|
|
|
|
|
|
public async Task<bool> CompleteLogin(string userId, AuthenticatorAssertionRawResponse response)
|
|
|
|
{
|
2021-04-20 07:06:32 +02:00
|
|
|
await using var dbContext = _contextFactory.CreateContext();
|
|
|
|
var user = await dbContext.Users.Include(applicationUser => applicationUser.Fido2Credentials)
|
|
|
|
.FirstOrDefaultAsync(applicationUser => applicationUser.Id == userId);
|
|
|
|
if (user == null || !LoginStore.TryGetValue(userId, out var options))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
var credential = user.Fido2Credentials
|
|
|
|
.Where(fido2Credential => fido2Credential.Type is Fido2Credential.CredentialType.FIDO2)
|
2021-11-11 13:03:08 +01:00
|
|
|
.Select(fido2Credential => (fido2Credential, fido2Credential.GetFido2Blob()))
|
2021-04-20 07:06:32 +02:00
|
|
|
.FirstOrDefault(fido2Credential => fido2Credential.Item2.Descriptor.Id.SequenceEqual(response.Id));
|
|
|
|
if (credential.Item2 is null)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 5. Make the assertion
|
|
|
|
var res = await _fido2.MakeAssertionAsync(response, options, credential.Item2.PublicKey,
|
2024-12-03 08:47:26 +01:00
|
|
|
credential.Item2.SignatureCounter, (x, cancellationToken) => Task.FromResult(true));
|
2021-04-20 07:06:32 +02:00
|
|
|
|
|
|
|
// 6. Store the updated counter
|
2021-12-31 08:59:02 +01:00
|
|
|
credential.Item2.SignatureCounter = res.Counter;
|
2021-04-20 07:06:32 +02:00
|
|
|
credential.fido2Credential.SetBlob(credential.Item2);
|
|
|
|
await dbContext.SaveChangesAsync();
|
|
|
|
LoginStore.Remove(userId, out _);
|
|
|
|
|
|
|
|
// 7. return OK to client
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|