mirror of
https://github.com/bitwarden/android.git
synced 2025-01-22 15:53:47 +03:00
314 lines
No EOL
14 KiB
C#
314 lines
No EOL
14 KiB
C#
using AuthenticationServices;
|
|
using Bit.App.Abstractions;
|
|
using Bit.App.Repositories;
|
|
using Bit.App.Resources;
|
|
using Bit.App.Services;
|
|
using Bit.iOS.Autofill.Models;
|
|
using Bit.iOS.Core;
|
|
using Bit.iOS.Core.Services;
|
|
using Bit.iOS.Core.Utilities;
|
|
using Foundation;
|
|
using Plugin.Connectivity;
|
|
using Plugin.Fingerprint;
|
|
using Plugin.Settings.Abstractions;
|
|
using SimpleInjector;
|
|
using System;
|
|
using UIKit;
|
|
using XLabs.Ioc;
|
|
using XLabs.Ioc.SimpleInjectorContainer;
|
|
|
|
namespace Bit.iOS.Autofill
|
|
{
|
|
public partial class CredentialProviderViewController : ASCredentialProviderViewController
|
|
{
|
|
private Context _context;
|
|
private bool _setupHockeyApp = false;
|
|
private IGoogleAnalyticsService _googleAnalyticsService;
|
|
|
|
public CredentialProviderViewController(IntPtr handle) : base(handle)
|
|
{ }
|
|
|
|
public override void ViewDidLoad()
|
|
{
|
|
SetIoc();
|
|
SetCulture();
|
|
base.ViewDidLoad();
|
|
_context = new Context();
|
|
_context.ExtContext = ExtensionContext;
|
|
_googleAnalyticsService = Resolver.Resolve<IGoogleAnalyticsService>();
|
|
|
|
if(!_setupHockeyApp)
|
|
{
|
|
var appIdService = Resolver.Resolve<IAppIdService>();
|
|
var crashManagerDelegate = new HockeyAppCrashManagerDelegate(appIdService, Resolver.Resolve<IAuthService>());
|
|
var manager = HockeyApp.iOS.BITHockeyManager.SharedHockeyManager;
|
|
manager.Configure("51f96ae568ba45f699a18ad9f63046c3", crashManagerDelegate);
|
|
manager.CrashManager.CrashManagerStatus = HockeyApp.iOS.BITCrashManagerStatus.AutoSend;
|
|
manager.UserId = appIdService.AppId;
|
|
manager.StartManager();
|
|
manager.Authenticator.AuthenticateInstallation();
|
|
_setupHockeyApp = true;
|
|
}
|
|
}
|
|
|
|
public override void PrepareCredentialList(ASCredentialServiceIdentifier[] serviceIdentifiers)
|
|
{
|
|
_context.ServiceIdentifiers = serviceIdentifiers;
|
|
_context.UrlString = serviceIdentifiers[0].Identifier;
|
|
base.PrepareCredentialList(serviceIdentifiers);
|
|
|
|
var authService = Resolver.Resolve<IAuthService>();
|
|
if(!authService.IsAuthenticated)
|
|
{
|
|
var alert = Dialogs.CreateAlert(null, AppResources.MustLogInMainApp, AppResources.Ok, (a) =>
|
|
{
|
|
CompleteRequest();
|
|
});
|
|
PresentViewController(alert, true, null);
|
|
return;
|
|
}
|
|
|
|
var lockService = Resolver.Resolve<ILockService>();
|
|
var lockType = lockService.GetLockTypeAsync(false).GetAwaiter().GetResult();
|
|
switch(lockType)
|
|
{
|
|
case App.Enums.LockType.Fingerprint:
|
|
PerformSegue("lockFingerprintSegue", this);
|
|
break;
|
|
case App.Enums.LockType.PIN:
|
|
PerformSegue("lockPinSegue", this);
|
|
break;
|
|
case App.Enums.LockType.Password:
|
|
PerformSegue("lockPasswordSegue", this);
|
|
break;
|
|
default:
|
|
PerformSegue("loginListSegue", this);
|
|
break;
|
|
}
|
|
}
|
|
|
|
public override void ProvideCredentialWithoutUserInteraction(ASPasswordCredentialIdentity credentialIdentity)
|
|
{
|
|
// base.ProvideCredentialWithoutUserInteraction(credentialIdentity);
|
|
|
|
bool canGetCredentials = false;
|
|
var authService = Resolver.Resolve<IAuthService>();
|
|
if(authService.IsAuthenticated)
|
|
{
|
|
var lockService = Resolver.Resolve<ILockService>();
|
|
var lockType = lockService.GetLockTypeAsync(false).GetAwaiter().GetResult();
|
|
canGetCredentials = lockType == App.Enums.LockType.None;
|
|
}
|
|
|
|
if(!canGetCredentials)
|
|
{
|
|
var err = new NSError(new NSString("ASExtensionErrorDomain"),
|
|
Convert.ToInt32(ASExtensionErrorCode.UserInteractionRequired), null);
|
|
this.ExtensionContext.CancelRequest(err);
|
|
return;
|
|
}
|
|
_context.CredentialIdentity = credentialIdentity;
|
|
ProvideCredential();
|
|
}
|
|
|
|
public override void PrepareInterfaceToProvideCredential(ASPasswordCredentialIdentity credentialIdentity)
|
|
{
|
|
var authService = Resolver.Resolve<IAuthService>();
|
|
if(!authService.IsAuthenticated)
|
|
{
|
|
var alert = Dialogs.CreateAlert(null, AppResources.MustLogInMainApp, AppResources.Ok, (a) =>
|
|
{
|
|
CompleteRequest();
|
|
});
|
|
PresentViewController(alert, true, null);
|
|
return;
|
|
}
|
|
|
|
_context.CredentialIdentity = credentialIdentity;
|
|
var lockService = Resolver.Resolve<ILockService>();
|
|
var lockType = lockService.GetLockTypeAsync(false).GetAwaiter().GetResult();
|
|
switch(lockType)
|
|
{
|
|
case App.Enums.LockType.Fingerprint:
|
|
PerformSegue("lockFingerprintSegue", this);
|
|
break;
|
|
case App.Enums.LockType.PIN:
|
|
PerformSegue("lockPinSegue", this);
|
|
break;
|
|
case App.Enums.LockType.Password:
|
|
PerformSegue("lockPasswordSegue", this);
|
|
break;
|
|
default:
|
|
ProvideCredential();
|
|
break;
|
|
}
|
|
}
|
|
|
|
public override void PrepareInterfaceForExtensionConfiguration()
|
|
{
|
|
base.PrepareInterfaceForExtensionConfiguration();
|
|
var authService = Resolver.Resolve<IAuthService>();
|
|
if (authService.IsAuthenticated)
|
|
{
|
|
var task = ASHelpers.ReplaceAllIdentities(Resolver.Resolve<ICipherService>());
|
|
}
|
|
ExtensionContext.CompleteExtensionConfigurationRequest();
|
|
}
|
|
|
|
public void CompleteRequest(string username = null, string password = null, string totp = null)
|
|
{
|
|
if(string.IsNullOrWhiteSpace(username) && string.IsNullOrWhiteSpace(password))
|
|
{
|
|
_googleAnalyticsService.TrackAutofillExtensionEvent("Canceled");
|
|
var err = new NSError(new NSString("ASExtensionErrorDomain"),
|
|
Convert.ToInt32(ASExtensionErrorCode.UserCanceled), null);
|
|
_googleAnalyticsService.Dispatch(() =>
|
|
{
|
|
NSRunLoop.Main.BeginInvokeOnMainThread(() =>
|
|
{
|
|
ExtensionContext.CancelRequest(err);
|
|
});
|
|
});
|
|
return;
|
|
}
|
|
|
|
if(!string.IsNullOrWhiteSpace(totp))
|
|
{
|
|
UIPasteboard.General.String = totp;
|
|
}
|
|
|
|
_googleAnalyticsService.TrackAutofillExtensionEvent("AutoFilled");
|
|
var cred = new ASPasswordCredential(username, password);
|
|
_googleAnalyticsService.Dispatch(() =>
|
|
{
|
|
NSRunLoop.Main.BeginInvokeOnMainThread(() =>
|
|
{
|
|
ExtensionContext.CompleteRequest(cred, null);
|
|
});
|
|
});
|
|
}
|
|
|
|
public override void PrepareForSegue(UIStoryboardSegue segue, NSObject sender)
|
|
{
|
|
var navController = segue.DestinationViewController as UINavigationController;
|
|
if(navController != null)
|
|
{
|
|
var listLoginController = navController.TopViewController as LoginListViewController;
|
|
var fingerprintViewController = navController.TopViewController as LockFingerprintViewController;
|
|
var pinViewController = navController.TopViewController as LockPinViewController;
|
|
var passwordViewController = navController.TopViewController as LockPasswordViewController;
|
|
|
|
if(listLoginController != null)
|
|
{
|
|
listLoginController.Context = _context;
|
|
listLoginController.CPViewController = this;
|
|
}
|
|
else if(fingerprintViewController != null)
|
|
{
|
|
fingerprintViewController.CPViewController = this;
|
|
}
|
|
else if(pinViewController != null)
|
|
{
|
|
pinViewController.CPViewController = this;
|
|
}
|
|
else if(passwordViewController != null)
|
|
{
|
|
passwordViewController.CPViewController = this;
|
|
}
|
|
}
|
|
}
|
|
|
|
public void DismissLockAndContinue()
|
|
{
|
|
DismissViewController(false, () =>
|
|
{
|
|
if(_context.CredentialIdentity != null)
|
|
{
|
|
ProvideCredential();
|
|
return;
|
|
}
|
|
PerformSegue("loginListSegue", this);
|
|
});
|
|
}
|
|
|
|
private void ProvideCredential()
|
|
{
|
|
var cipherService = Resolver.Resolve<ICipherService>();
|
|
var cipher = cipherService.GetByIdAsync(_context.CredentialIdentity.RecordIdentifier).GetAwaiter().GetResult();
|
|
if(cipher == null || cipher.Type != App.Enums.CipherType.Login)
|
|
{
|
|
var err = new NSError(new NSString("ASExtensionErrorDomain"),
|
|
Convert.ToInt32(ASExtensionErrorCode.CredentialIdentityNotFound), null);
|
|
ExtensionContext.CancelRequest(err);
|
|
return;
|
|
}
|
|
|
|
CompleteRequest(cipher.Login.Username?.Decrypt(cipher.OrganizationId),
|
|
cipher.Login.Password?.Decrypt(cipher.OrganizationId),
|
|
cipher.Login.Totp?.Decrypt(cipher.OrganizationId));
|
|
}
|
|
|
|
private void SetIoc()
|
|
{
|
|
var container = new Container();
|
|
|
|
// Services
|
|
container.RegisterSingleton<IDatabaseService, DatabaseService>();
|
|
container.RegisterSingleton<ISqlService, SqlService>();
|
|
container.RegisterSingleton<ISecureStorageService, KeyChainStorageService>();
|
|
container.RegisterSingleton<ICryptoService, CryptoService>();
|
|
container.RegisterSingleton<IKeyDerivationService, CommonCryptoKeyDerivationService>();
|
|
container.RegisterSingleton<IAuthService, AuthService>();
|
|
container.RegisterSingleton<IFolderService, FolderService>();
|
|
container.RegisterSingleton<ICollectionService, CollectionService>();
|
|
container.RegisterSingleton<ICipherService, CipherService>();
|
|
container.RegisterSingleton<ISyncService, SyncService>();
|
|
container.RegisterSingleton<IDeviceActionService, NoopDeviceActionService>();
|
|
container.RegisterSingleton<IAppIdService, AppIdService>();
|
|
container.RegisterSingleton<IPasswordGenerationService, PasswordGenerationService>();
|
|
container.RegisterSingleton<ILockService, LockService>();
|
|
container.RegisterSingleton<IAppInfoService, AppInfoService>();
|
|
container.RegisterSingleton<IGoogleAnalyticsService, GoogleAnalyticsService>();
|
|
container.RegisterSingleton<IDeviceInfoService, DeviceInfoService>();
|
|
container.RegisterSingleton<ILocalizeService, LocalizeService>();
|
|
container.RegisterSingleton<ILogService, LogService>();
|
|
container.RegisterSingleton<IHttpService, HttpService>();
|
|
container.RegisterSingleton<ITokenService, TokenService>();
|
|
container.RegisterSingleton<ISettingsService, SettingsService>();
|
|
container.RegisterSingleton<IAppSettingsService, AppSettingsService>();
|
|
|
|
// Repositories
|
|
container.RegisterSingleton<IFolderRepository, FolderRepository>();
|
|
container.RegisterSingleton<IFolderApiRepository, FolderApiRepository>();
|
|
container.RegisterSingleton<ICipherRepository, CipherRepository>();
|
|
container.RegisterSingleton<IAttachmentRepository, AttachmentRepository>();
|
|
container.RegisterSingleton<IConnectApiRepository, ConnectApiRepository>();
|
|
container.RegisterSingleton<IDeviceApiRepository, DeviceApiRepository>();
|
|
container.RegisterSingleton<IAccountsApiRepository, AccountsApiRepository>();
|
|
container.RegisterSingleton<ICipherApiRepository, CipherApiRepository>();
|
|
container.RegisterSingleton<ISettingsRepository, SettingsRepository>();
|
|
container.RegisterSingleton<ISettingsApiRepository, SettingsApiRepository>();
|
|
container.RegisterSingleton<ITwoFactorApiRepository, TwoFactorApiRepository>();
|
|
container.RegisterSingleton<ISyncApiRepository, SyncApiRepository>();
|
|
container.RegisterSingleton<ICollectionRepository, CollectionRepository>();
|
|
container.RegisterSingleton<ICipherCollectionRepository, CipherCollectionRepository>();
|
|
|
|
// Other
|
|
container.RegisterSingleton(CrossConnectivity.Current);
|
|
container.RegisterSingleton(CrossFingerprint.Current);
|
|
|
|
var settings = new Settings("group.com.8bit.bitwarden");
|
|
container.RegisterSingleton<ISettings>(settings);
|
|
|
|
Resolver.ResetResolver(new SimpleInjectorResolver(container));
|
|
}
|
|
|
|
private void SetCulture()
|
|
{
|
|
var localizeService = Resolver.Resolve<ILocalizeService>();
|
|
var ci = localizeService.GetCurrentCultureInfo();
|
|
AppResources.Culture = ci;
|
|
localizeService.SetLocale(ci);
|
|
}
|
|
}
|
|
} |