mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
Add distributed cache support
This commit is contained in:
parent
b03e3c3b8c
commit
ce003e8efc
@ -21,6 +21,9 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
private readonly IDistributedCache _distributedCache;
|
private readonly IDistributedCache _distributedCache;
|
||||||
private readonly IUserRepository _userRepository;
|
private readonly IUserRepository _userRepository;
|
||||||
|
|
||||||
|
const string REGISTER_SESSION_KEY = "opaque_register_session_{0}";
|
||||||
|
const string LOGIN_SESSION_KEY = "opaque_login_session_{0}";
|
||||||
|
|
||||||
public OpaqueKeyExchangeService(
|
public OpaqueKeyExchangeService(
|
||||||
IOpaqueKeyExchangeCredentialRepository opaqueKeyExchangeCredentialRepository,
|
IOpaqueKeyExchangeCredentialRepository opaqueKeyExchangeCredentialRepository,
|
||||||
IDistributedCache distributedCache,
|
IDistributedCache distributedCache,
|
||||||
@ -35,36 +38,36 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
|
|
||||||
public async Task<OpaqueRegistrationStartResponse> StartRegistration(byte[] request, User user, Models.Api.Request.Opaque.CipherConfiguration cipherConfiguration)
|
public async Task<OpaqueRegistrationStartResponse> StartRegistration(byte[] request, User user, Models.Api.Request.Opaque.CipherConfiguration cipherConfiguration)
|
||||||
{
|
{
|
||||||
return await Task.Run(() =>
|
var registrationRequest = _bitwardenOpaque.StartRegistration(cipherConfiguration.ToNativeConfiguration(), null, request, user.Id.ToString());
|
||||||
{
|
|
||||||
var registrationRequest = _bitwardenOpaque.StartRegistration(cipherConfiguration.ToNativeConfiguration(), null, request, user.Id.ToString());
|
var sessionId = Guid.NewGuid();
|
||||||
var registrationReseponse = registrationRequest.registrationResponse;
|
var registerSession = new RegisterSession() { SessionId = sessionId, ServerSetup = registrationRequest.serverSetup, CipherConfiguration = cipherConfiguration, UserId = user.Id };
|
||||||
var serverSetup = registrationRequest.serverSetup;
|
await _distributedCache.SetAsync(string.Format(REGISTER_SESSION_KEY, sessionId), Encoding.ASCII.GetBytes(JsonSerializer.Serialize(registerSession)));
|
||||||
// persist server setup
|
|
||||||
var sessionId = Guid.NewGuid();
|
return new OpaqueRegistrationStartResponse(sessionId, Convert.ToBase64String(registrationRequest.registrationResponse));
|
||||||
SessionStore.RegisterSessions.Add(sessionId, new RegisterSession() { SessionId = sessionId, ServerSetup = serverSetup, CipherConfiguration = cipherConfiguration, UserId = user.Id });
|
|
||||||
return new OpaqueRegistrationStartResponse(sessionId, Convert.ToBase64String(registrationReseponse));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task FinishRegistration(Guid sessionId, byte[] registrationUpload, User user, RotateableOpaqueKeyset keyset)
|
public async Task FinishRegistration(Guid sessionId, byte[] registrationUpload, User user, RotateableOpaqueKeyset keyset)
|
||||||
{
|
{
|
||||||
await Task.Run(() =>
|
var serializedRegisterSession = await _distributedCache.GetAsync(string.Format(REGISTER_SESSION_KEY, sessionId));
|
||||||
|
if (serializedRegisterSession == null)
|
||||||
{
|
{
|
||||||
var cipherConfiguration = SessionStore.RegisterSessions[sessionId].CipherConfiguration;
|
throw new InvalidOperationException("Session not found");
|
||||||
try
|
}
|
||||||
{
|
|
||||||
var registrationFinish = _bitwardenOpaque.FinishRegistration(cipherConfiguration.ToNativeConfiguration(), registrationUpload);
|
|
||||||
SessionStore.RegisterSessions[sessionId].PasswordFile = registrationFinish.serverRegistration;
|
|
||||||
|
|
||||||
SessionStore.RegisterSessions[sessionId].KeySet = Encoding.ASCII.GetBytes(JsonSerializer.Serialize(keyset));
|
try
|
||||||
}
|
{
|
||||||
catch (Exception e)
|
var registerSession = JsonSerializer.Deserialize<RegisterSession>(Encoding.ASCII.GetString(serializedRegisterSession))!;
|
||||||
{
|
var registrationFinish = _bitwardenOpaque.FinishRegistration(registerSession.CipherConfiguration.ToNativeConfiguration(), registrationUpload);
|
||||||
SessionStore.RegisterSessions.Remove(sessionId);
|
registerSession.PasswordFile = registrationFinish.serverRegistration;
|
||||||
throw new Exception(e.Message);
|
registerSession.KeySet = Encoding.ASCII.GetBytes(JsonSerializer.Serialize(keyset));
|
||||||
}
|
await _distributedCache.SetAsync(string.Format(REGISTER_SESSION_KEY, sessionId), Encoding.ASCII.GetBytes(JsonSerializer.Serialize(registerSession)));
|
||||||
});
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
await _distributedCache.RemoveAsync(string.Format(REGISTER_SESSION_KEY, sessionId));
|
||||||
|
throw new Exception(e.Message);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<(Guid, byte[])> StartLogin(byte[] request, string email)
|
public async Task<(Guid, byte[])> StartLogin(byte[] request, string email)
|
||||||
@ -90,48 +93,51 @@ public class OpaqueKeyExchangeService : IOpaqueKeyExchangeService
|
|||||||
|
|
||||||
var loginResponse = _bitwardenOpaque.StartLogin(cipherConfiguration.ToNativeConfiguration(), serverSetup, serverRegistration, request, user.Id.ToString());
|
var loginResponse = _bitwardenOpaque.StartLogin(cipherConfiguration.ToNativeConfiguration(), serverSetup, serverRegistration, request, user.Id.ToString());
|
||||||
var sessionId = Guid.NewGuid();
|
var sessionId = Guid.NewGuid();
|
||||||
SessionStore.LoginSessions.Add(sessionId, new LoginSession() { UserId = user.Id, LoginState = loginResponse.state, CipherConfiguration = cipherConfiguration });
|
var loginSession = new LoginSession() { UserId = user.Id, LoginState = loginResponse.state, CipherConfiguration = cipherConfiguration };
|
||||||
|
await _distributedCache.SetAsync(string.Format(LOGIN_SESSION_KEY, sessionId), Encoding.ASCII.GetBytes(JsonSerializer.Serialize(loginSession)));
|
||||||
return (sessionId, loginResponse.credentialResponse);
|
return (sessionId, loginResponse.credentialResponse);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<bool> FinishLogin(Guid sessionId, byte[] credentialFinalization)
|
public async Task<bool> FinishLogin(Guid sessionId, byte[] credentialFinalization)
|
||||||
{
|
{
|
||||||
return await Task.Run(async () =>
|
var serializedLoginSession = await _distributedCache.GetAsync(string.Format(LOGIN_SESSION_KEY, sessionId));
|
||||||
|
if (serializedLoginSession == null)
|
||||||
{
|
{
|
||||||
if (!SessionStore.LoginSessions.ContainsKey(sessionId))
|
throw new InvalidOperationException("Session not found");
|
||||||
{
|
}
|
||||||
throw new InvalidOperationException("Session not found");
|
var loginSession = JsonSerializer.Deserialize<LoginSession>(Encoding.ASCII.GetString(serializedLoginSession))!;
|
||||||
}
|
|
||||||
|
|
||||||
var userId = SessionStore.LoginSessions[sessionId].UserId;
|
var credential = await _opaqueKeyExchangeCredentialRepository.GetByUserIdAsync(loginSession.UserId);
|
||||||
var credential = await _opaqueKeyExchangeCredentialRepository.GetByUserIdAsync(userId);
|
if (credential == null)
|
||||||
if (credential == null)
|
{
|
||||||
{
|
throw new InvalidOperationException("Credential not found");
|
||||||
throw new InvalidOperationException("Credential not found");
|
}
|
||||||
}
|
|
||||||
|
|
||||||
var loginState = SessionStore.LoginSessions[sessionId].LoginState;
|
var loginState = loginSession.LoginState;
|
||||||
var cipherConfiguration = SessionStore.LoginSessions[sessionId].CipherConfiguration;
|
var cipherConfiguration = loginSession.CipherConfiguration;
|
||||||
SessionStore.LoginSessions.Remove(sessionId);
|
await _distributedCache.RemoveAsync(string.Format(LOGIN_SESSION_KEY, sessionId));
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var success = _bitwardenOpaque.FinishLogin(cipherConfiguration.ToNativeConfiguration(), loginState, credentialFinalization);
|
var success = _bitwardenOpaque.FinishLogin(cipherConfiguration.ToNativeConfiguration(), loginState, credentialFinalization);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
// print
|
// print
|
||||||
Console.WriteLine(e.Message);
|
Console.WriteLine(e.Message);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task SetActive(Guid sessionId, User user)
|
public async Task SetActive(Guid sessionId, User user)
|
||||||
{
|
{
|
||||||
var session = SessionStore.RegisterSessions[sessionId];
|
var serializedRegisterSession = await _distributedCache.GetAsync(string.Format(REGISTER_SESSION_KEY, sessionId));
|
||||||
SessionStore.RegisterSessions.Remove(sessionId);
|
if (serializedRegisterSession == null)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException("Session not found");
|
||||||
|
}
|
||||||
|
var session = JsonSerializer.Deserialize<RegisterSession>(Encoding.ASCII.GetString(serializedRegisterSession))!;
|
||||||
|
|
||||||
if (session.UserId != user.Id)
|
if (session.UserId != user.Id)
|
||||||
{
|
{
|
||||||
@ -194,9 +200,3 @@ public class LoginSession
|
|||||||
public required byte[] LoginState { get; set; }
|
public required byte[] LoginState { get; set; }
|
||||||
public required Models.Api.Request.Opaque.CipherConfiguration CipherConfiguration { get; set; }
|
public required Models.Api.Request.Opaque.CipherConfiguration CipherConfiguration { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class SessionStore()
|
|
||||||
{
|
|
||||||
public static Dictionary<Guid, RegisterSession> RegisterSessions = new Dictionary<Guid, RegisterSession>();
|
|
||||||
public static Dictionary<Guid, LoginSession> LoginSessions = new Dictionary<Guid, LoginSession>();
|
|
||||||
}
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user