mirror of
https://github.com/bitwarden/android.git
synced 2024-12-18 23:31:52 +03:00
two-factor login re-worked with new auth flow
This commit is contained in:
parent
54f8771a9c
commit
d112e0ea42
4 changed files with 52 additions and 13 deletions
|
@ -1,4 +1,5 @@
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Bit.App.Models.Api
|
namespace Bit.App.Models.Api
|
||||||
{
|
{
|
||||||
|
@ -12,5 +13,6 @@ namespace Bit.App.Models.Api
|
||||||
public string RefreshToken { get; set; }
|
public string RefreshToken { get; set; }
|
||||||
[JsonProperty("token_type")]
|
[JsonProperty("token_type")]
|
||||||
public string TokenType { get; set; }
|
public string TokenType { get; set; }
|
||||||
|
public List<int> TwoFactorProviders { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -197,6 +197,13 @@ namespace Bit.App.Pages
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(response.Result.TwoFactorProviders != null && response.Result.TwoFactorProviders.Count > 0)
|
||||||
|
{
|
||||||
|
_googleAnalyticsService.TrackAppEvent("LoggedIn To Two-step");
|
||||||
|
await Navigation.PushAsync(new LoginTwoFactorPage(request.Email, request.MasterPasswordHash, key));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
_cryptoService.Key = key;
|
_cryptoService.Key = key;
|
||||||
_tokenService.Token = response.Result.AccessToken;
|
_tokenService.Token = response.Result.AccessToken;
|
||||||
_tokenService.RefreshToken = response.Result.RefreshToken;
|
_tokenService.RefreshToken = response.Result.RefreshToken;
|
||||||
|
@ -211,15 +218,8 @@ namespace Bit.App.Pages
|
||||||
_pushNotification.Register();
|
_pushNotification.Register();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(false) // TODO: 2FA
|
var task = Task.Run(async () => await _syncService.FullSyncAsync());
|
||||||
{
|
Application.Current.MainPage = new MainPage();
|
||||||
await Navigation.PushAsync(new LoginTwoFactorPage());
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var task = Task.Run(async () => await _syncService.FullSyncAsync());
|
|
||||||
Application.Current.MainPage = new MainPage();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,8 @@ using Xamarin.Forms;
|
||||||
using XLabs.Ioc;
|
using XLabs.Ioc;
|
||||||
using Acr.UserDialogs;
|
using Acr.UserDialogs;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
using Plugin.Settings.Abstractions;
|
||||||
|
using PushNotification.Plugin.Abstractions;
|
||||||
|
|
||||||
namespace Bit.App.Pages
|
namespace Bit.App.Pages
|
||||||
{
|
{
|
||||||
|
@ -20,10 +22,20 @@ namespace Bit.App.Pages
|
||||||
private IAppIdService _appIdService;
|
private IAppIdService _appIdService;
|
||||||
private IUserDialogs _userDialogs;
|
private IUserDialogs _userDialogs;
|
||||||
private ISyncService _syncService;
|
private ISyncService _syncService;
|
||||||
|
private ISettings _settings;
|
||||||
|
private IGoogleAnalyticsService _googleAnalyticsService;
|
||||||
|
private IPushNotification _pushNotification;
|
||||||
|
private readonly string _email;
|
||||||
|
private readonly string _masterPasswordHash;
|
||||||
|
private readonly byte[] _key;
|
||||||
|
|
||||||
public LoginTwoFactorPage()
|
public LoginTwoFactorPage(string email, string masterPasswordHash, byte[] key)
|
||||||
: base(updateActivity: false)
|
: base(updateActivity: false)
|
||||||
{
|
{
|
||||||
|
_email = email;
|
||||||
|
_masterPasswordHash = masterPasswordHash;
|
||||||
|
_key = key;
|
||||||
|
|
||||||
_cryptoService = Resolver.Resolve<ICryptoService>();
|
_cryptoService = Resolver.Resolve<ICryptoService>();
|
||||||
_authService = Resolver.Resolve<IAuthService>();
|
_authService = Resolver.Resolve<IAuthService>();
|
||||||
_tokenService = Resolver.Resolve<ITokenService>();
|
_tokenService = Resolver.Resolve<ITokenService>();
|
||||||
|
@ -31,6 +43,9 @@ namespace Bit.App.Pages
|
||||||
_appIdService = Resolver.Resolve<IAppIdService>();
|
_appIdService = Resolver.Resolve<IAppIdService>();
|
||||||
_userDialogs = Resolver.Resolve<IUserDialogs>();
|
_userDialogs = Resolver.Resolve<IUserDialogs>();
|
||||||
_syncService = Resolver.Resolve<ISyncService>();
|
_syncService = Resolver.Resolve<ISyncService>();
|
||||||
|
_settings = Resolver.Resolve<ISettings>();
|
||||||
|
_googleAnalyticsService = Resolver.Resolve<IGoogleAnalyticsService>();
|
||||||
|
_pushNotification = Resolver.Resolve<IPushNotification>();
|
||||||
|
|
||||||
Init();
|
Init();
|
||||||
}
|
}
|
||||||
|
@ -138,9 +153,10 @@ namespace Bit.App.Pages
|
||||||
|
|
||||||
var request = new TokenRequest
|
var request = new TokenRequest
|
||||||
{
|
{
|
||||||
// TODO: username and pass from previous page
|
Email = _email,
|
||||||
|
MasterPasswordHash = _masterPasswordHash,
|
||||||
Token = CodeCell.Entry.Text.Replace(" ", ""),
|
Token = CodeCell.Entry.Text.Replace(" ", ""),
|
||||||
Provider = 0,
|
Provider = 0, // Authenticator app (only 1 provider for now, so hard coded)
|
||||||
Device = new DeviceRequest(_appIdService, _deviceInfoService)
|
Device = new DeviceRequest(_appIdService, _deviceInfoService)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -153,10 +169,19 @@ namespace Bit.App.Pages
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_cryptoService.Key = _key;
|
||||||
_tokenService.Token = response.Result.AccessToken;
|
_tokenService.Token = response.Result.AccessToken;
|
||||||
_tokenService.RefreshToken = response.Result.RefreshToken;
|
_tokenService.RefreshToken = response.Result.RefreshToken;
|
||||||
_authService.UserId = _tokenService.TokenUserId;
|
_authService.UserId = _tokenService.TokenUserId;
|
||||||
_authService.Email = _tokenService.TokenEmail;
|
_authService.Email = _tokenService.TokenEmail;
|
||||||
|
_settings.AddOrUpdateValue(Constants.LastLoginEmail, _authService.Email);
|
||||||
|
_googleAnalyticsService.RefreshUserId();
|
||||||
|
_googleAnalyticsService.TrackAppEvent("LoggedIn From Two-step");
|
||||||
|
|
||||||
|
if(Device.OS == TargetPlatform.Android)
|
||||||
|
{
|
||||||
|
_pushNotification.Register();
|
||||||
|
}
|
||||||
|
|
||||||
var task = Task.Run(async () => await _syncService.FullSyncAsync());
|
var task = Task.Run(async () => await _syncService.FullSyncAsync());
|
||||||
Application.Current.MainPage = new MainPage();
|
Application.Current.MainPage = new MainPage();
|
||||||
|
|
|
@ -6,6 +6,8 @@ using Bit.App.Models.Api;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using Plugin.Connectivity.Abstractions;
|
using Plugin.Connectivity.Abstractions;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Bit.App.Repositories
|
namespace Bit.App.Repositories
|
||||||
{
|
{
|
||||||
|
@ -39,12 +41,22 @@ namespace Bit.App.Repositories
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var response = await client.SendAsync(requestMessage).ConfigureAwait(false);
|
var response = await client.SendAsync(requestMessage).ConfigureAwait(false);
|
||||||
|
var responseContent = await response.Content.ReadAsStringAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
if(!response.IsSuccessStatusCode)
|
if(!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
|
var errorResponse = JObject.Parse(responseContent);
|
||||||
|
if(errorResponse["TwoFactorProviders"] != null)
|
||||||
|
{
|
||||||
|
return ApiResult<TokenResponse>.Success(new TokenResponse
|
||||||
|
{
|
||||||
|
TwoFactorProviders = errorResponse["TwoFactorProviders"].ToObject<List<int>>()
|
||||||
|
}, response.StatusCode);
|
||||||
|
}
|
||||||
|
|
||||||
return await HandleErrorAsync<TokenResponse>(response).ConfigureAwait(false);
|
return await HandleErrorAsync<TokenResponse>(response).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
var responseContent = await response.Content.ReadAsStringAsync().ConfigureAwait(false);
|
|
||||||
var responseObj = JsonConvert.DeserializeObject<TokenResponse>(responseContent);
|
var responseObj = JsonConvert.DeserializeObject<TokenResponse>(responseContent);
|
||||||
return ApiResult<TokenResponse>.Success(responseObj, response.StatusCode);
|
return ApiResult<TokenResponse>.Success(responseObj, response.StatusCode);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue