mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
Fake responses for non-existent users (#5538)
This commit is contained in:
parent
5016ece4ff
commit
77206b12a9
@ -14,7 +14,7 @@ public class OpaqueRegistrationStartRequest
|
|||||||
|
|
||||||
public class OpaqueKeyExchangeCipherConfiguration
|
public class OpaqueKeyExchangeCipherConfiguration
|
||||||
{
|
{
|
||||||
static string OpaqueKe3Ristretto3DHArgonSuite = "OPAQUE_3_RISTRETTO255_OPRF_RISTRETTO255_KEGROUP_3DH_KEX_ARGON2ID13_KSF";
|
public static string OpaqueKe3Ristretto3DHArgonSuite = "OPAQUE_3_RISTRETTO255_OPRF_RISTRETTO255_KEGROUP_3DH_KEX_ARGON2ID13_KSF";
|
||||||
|
|
||||||
[Required]
|
[Required]
|
||||||
public string CipherSuite { get; set; }
|
public string CipherSuite { get; set; }
|
||||||
|
@ -8,6 +8,7 @@ using Bit.Core.Auth.Repositories;
|
|||||||
using Bit.Core.Auth.Utilities;
|
using Bit.Core.Auth.Utilities;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Core.Settings;
|
||||||
using Bit.Core.Utilities;
|
using Bit.Core.Utilities;
|
||||||
using Bitwarden.Opaque;
|
using Bitwarden.Opaque;
|
||||||
using Microsoft.Extensions.Caching.Distributed;
|
using Microsoft.Extensions.Caching.Distributed;
|
||||||
@ -25,6 +26,7 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
private readonly IUserRepository _userRepository;
|
private readonly IUserRepository _userRepository;
|
||||||
private readonly ILogger<OpaqueKeyExchangeService> _logger;
|
private readonly ILogger<OpaqueKeyExchangeService> _logger;
|
||||||
private readonly DistributedCacheEntryOptions _distributedCacheEntryOptions;
|
private readonly DistributedCacheEntryOptions _distributedCacheEntryOptions;
|
||||||
|
private readonly byte[] _defaultKdfHmacKey = null;
|
||||||
|
|
||||||
const string REGISTRATION_SESSION_KEY = "opaque_register_session_{0}";
|
const string REGISTRATION_SESSION_KEY = "opaque_register_session_{0}";
|
||||||
const string LOGIN_SESSION_KEY = "opaque_login_session_{0}";
|
const string LOGIN_SESSION_KEY = "opaque_login_session_{0}";
|
||||||
@ -33,7 +35,8 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
IOpaqueKeyExchangeCredentialRepository opaqueKeyExchangeCredentialRepository,
|
IOpaqueKeyExchangeCredentialRepository opaqueKeyExchangeCredentialRepository,
|
||||||
IDistributedCache distributedCache,
|
IDistributedCache distributedCache,
|
||||||
IUserRepository userRepository,
|
IUserRepository userRepository,
|
||||||
ILogger<OpaqueKeyExchangeService> logger
|
ILogger<OpaqueKeyExchangeService> logger,
|
||||||
|
GlobalSettings globalSettings
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
_bitwardenOpaque = new BitwardenOpaqueServer();
|
_bitwardenOpaque = new BitwardenOpaqueServer();
|
||||||
@ -45,6 +48,14 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
{
|
{
|
||||||
AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(1)
|
AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(1)
|
||||||
};
|
};
|
||||||
|
if (CoreHelpers.SettingHasValue(globalSettings.KdfDefaultHashKey))
|
||||||
|
{
|
||||||
|
_defaultKdfHmacKey = Encoding.UTF8.GetBytes(globalSettings.KdfDefaultHashKey);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_defaultKdfHmacKey = new byte[32];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<OpaqueRegistrationStartResponse> StartRegistration(
|
public async Task<OpaqueRegistrationStartResponse> StartRegistration(
|
||||||
@ -110,18 +121,39 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
// todo: don't allow user enumeration
|
var user = await _userRepository.GetByEmailAsync(email);
|
||||||
var user = await _userRepository.GetByEmailAsync(email)
|
// Fake user to prevent user enumeration
|
||||||
?? throw new InvalidOperationException("User not found");
|
user ??= new User() { Id = Guid.Empty };
|
||||||
|
|
||||||
// todo: generate fake credential
|
byte[] serverSetup = null;
|
||||||
var credential = await _opaqueKeyExchangeCredentialRepository.GetByUserIdAsync(user.Id)
|
byte[] serverRegistration = null;
|
||||||
?? throw new InvalidOperationException("Credential not found");
|
OpaqueKeyExchangeCipherConfiguration cipherConfiguration;
|
||||||
|
var credential = await _opaqueKeyExchangeCredentialRepository.GetByUserIdAsync(user.Id);
|
||||||
var cipherConfiguration = JsonSerializer.Deserialize<OpaqueKeyExchangeCipherConfiguration>(credential.CipherConfiguration)!;
|
if (credential != null)
|
||||||
var credentialBlob = JsonSerializer.Deserialize<OpaqueKeyExchangeCredentialBlob>(credential.CredentialBlob)!;
|
{
|
||||||
var serverSetup = credentialBlob.ServerSetup;
|
cipherConfiguration = JsonSerializer.Deserialize<OpaqueKeyExchangeCipherConfiguration>(credential.CipherConfiguration)!;
|
||||||
var serverRegistration = credentialBlob.PasswordFile;
|
var credentialBlob = JsonSerializer.Deserialize<OpaqueKeyExchangeCredentialBlob>(credential.CredentialBlob)!;
|
||||||
|
serverSetup = credentialBlob.ServerSetup;
|
||||||
|
serverRegistration = credentialBlob.PasswordFile;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Generate a fake registration for non-existent users
|
||||||
|
cipherConfiguration = new OpaqueKeyExchangeCipherConfiguration()
|
||||||
|
{
|
||||||
|
CipherSuite = OpaqueKeyExchangeCipherConfiguration.OpaqueKe3Ristretto3DHArgonSuite,
|
||||||
|
Argon2Parameters = new Argon2KsfParameters()
|
||||||
|
{
|
||||||
|
Memory = 0,
|
||||||
|
Iterations = 0,
|
||||||
|
Parallelism = 0
|
||||||
|
}
|
||||||
|
};
|
||||||
|
var hmacMessage = Encoding.UTF8.GetBytes(email.Trim().ToLowerInvariant());
|
||||||
|
using var hmac = new System.Security.Cryptography.HMACSHA256(_defaultKdfHmacKey);
|
||||||
|
var hmacHash = hmac.ComputeHash(hmacMessage);
|
||||||
|
(serverSetup, serverRegistration) = _bitwardenOpaque.SeededFakeRegistration(hmacHash);
|
||||||
|
}
|
||||||
|
|
||||||
var loginResponse = _bitwardenOpaque.StartLogin(
|
var loginResponse = _bitwardenOpaque.StartLogin(
|
||||||
cipherConfiguration.ToNativeConfiguration(), serverSetup, serverRegistration, request, user.Id.ToString());
|
cipherConfiguration.ToNativeConfiguration(), serverSetup, serverRegistration, request, user.Id.ToString());
|
||||||
@ -156,9 +188,6 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
?? throw new InvalidOperationException("Session not found");
|
?? throw new InvalidOperationException("Session not found");
|
||||||
var loginSession = JsonSerializer.Deserialize<OpaqueKeyExchangeLoginSession>(Encoding.ASCII.GetString(serializedLoginSession))!;
|
var loginSession = JsonSerializer.Deserialize<OpaqueKeyExchangeLoginSession>(Encoding.ASCII.GetString(serializedLoginSession))!;
|
||||||
|
|
||||||
var credential = await _opaqueKeyExchangeCredentialRepository.GetByUserIdAsync(loginSession.UserId)
|
|
||||||
?? throw new InvalidOperationException("Credential not found");
|
|
||||||
|
|
||||||
var loginState = loginSession.LoginState;
|
var loginState = loginSession.LoginState;
|
||||||
var cipherConfiguration = loginSession.CipherConfiguration;
|
var cipherConfiguration = loginSession.CipherConfiguration;
|
||||||
await ClearAuthenticationSession(sessionId);
|
await ClearAuthenticationSession(sessionId);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user