diff --git a/src/Android/AutofillService.cs b/src/Android/AutofillService.cs deleted file mode 100644 index 1a2bc46c4..000000000 --- a/src/Android/AutofillService.cs +++ /dev/null @@ -1,549 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using Android.AccessibilityServices; -using Android.App; -using Android.Content; -using Android.OS; -using Android.Views.Accessibility; -using Bit.App.Abstractions; -using XLabs.Ioc; -using Bit.App.Resources; - -namespace Bit.Android -{ - [Service(Permission = global::Android.Manifest.Permission.BindAccessibilityService, Label = "Bitwarden")] - [IntentFilter(new string[] { "android.accessibilityservice.AccessibilityService" })] - [MetaData("android.accessibilityservice", Resource = "@xml/accessibilityservice")] - public class AutofillService : AccessibilityService - { - private NotificationChannel _notificationChannel; - - private const string BitwardenTag = "bw_access"; - private const int AutoFillNotificationId = 34573; - private const string SystemUiPackage = "com.android.systemui"; - private const string BitwardenPackage = "com.x8bit.bitwarden"; - private const string BitwardenWebsite = "vault.bitwarden.com"; - - private static Dictionary SupportedBrowsers => new List - { - new Browser("com.android.chrome", "url_bar"), - new Browser("com.chrome.beta", "url_bar"), - new Browser("org.chromium.chrome", "url_bar"), - new Browser("com.android.browser", "url"), - new Browser("com.brave.browser", "url_bar"), - new Browser("com.opera.browser", "url_field"), - new Browser("com.opera.browser.beta", "url_field"), - new Browser("com.opera.mini.native", "url_field"), - new Browser("com.opera.touch", "addressbarEdit"), - new Browser("com.chrome.dev", "url_bar"), - new Browser("com.chrome.canary", "url_bar"), - new Browser("com.google.android.apps.chrome", "url_bar"), - new Browser("com.google.android.apps.chrome_dev", "url_bar"), - new Browser("org.codeaurora.swe.browser", "url_bar"), - new Browser("org.iron.srware", "url_bar"), - new Browser("com.sec.android.app.sbrowser", "location_bar_edit_text"), - new Browser("com.sec.android.app.sbrowser.beta", "location_bar_edit_text"), - new Browser("com.yandex.browser", "bro_omnibar_address_title_text", - (s) => s.Split(new char[]{' ', ' '}).FirstOrDefault()), // 0 = Regular Space, 1 = No-break space (00A0) - new Browser("org.mozilla.firefox", "url_bar_title"), - new Browser("org.mozilla.firefox_beta", "url_bar_title"), - new Browser("org.mozilla.fennec_aurora", "url_bar_title"), - new Browser("org.mozilla.fennec_fdroid", "url_bar_title"), - new Browser("org.mozilla.focus", "display_url"), - new Browser("org.mozilla.klar", "display_url"), - new Browser("org.mozilla.fenix", "mozac_browser_toolbar_url_view"), - new Browser("org.mozilla.reference.browser", "mozac_browser_toolbar_url_view"), - new Browser("com.ghostery.android.ghostery", "search_field"), - new Browser("org.adblockplus.browser", "url_bar_title"), - new Browser("com.htc.sense.browser", "title"), - new Browser("com.amazon.cloud9", "url"), - new Browser("mobi.mgeek.TunnyBrowser", "title"), - new Browser("com.nubelacorp.javelin", "enterUrl"), - new Browser("com.jerky.browser2", "enterUrl"), - new Browser("com.mx.browser", "address_editor_with_progress"), - new Browser("com.mx.browser.tablet", "address_editor_with_progress"), - new Browser("com.linkbubble.playstore", "url_text"), - new Browser("com.ksmobile.cb", "address_bar_edit_text"), - new Browser("acr.browser.lightning", "search"), - new Browser("acr.browser.barebones", "search"), - new Browser("com.microsoft.emmx", "url_bar"), - new Browser("com.duckduckgo.mobile.android", "omnibarTextInput"), - new Browser("mark.via.gp", "aw"), - new Browser("org.bromite.bromite", "url_bar"), - new Browser("com.kiwibrowser.browser", "url_bar"), - new Browser("com.ecosia.android", "url_bar"), - new Browser("com.qwant.liberty", "url_bar_title"), - }.ToDictionary(n => n.PackageName); - - // Known packages to skip - private static HashSet FilteredPackageNames => new HashSet - { - SystemUiPackage, - "com.google.android.googlequicksearchbox", - "com.google.android.apps.nexuslauncher", - "com.google.android.launcher", - "com.computer.desktop.ui.launcher", - "com.launcher.notelauncher", - "com.anddoes.launcher", - "com.actionlauncher.playstore", - "ch.deletescape.lawnchair.plah", - "com.microsoft.launcher", - "com.teslacoilsw.launcher", - "com.teslacoilsw.launcher.prime", - "is.shortcut", - "me.craftsapp.nlauncher", - "com.ss.squarehome2" - }; - - private readonly IAppSettingsService _appSettings; - private long _lastNotificationTime = 0; - private string _lastNotificationUri = null; - private HashSet _launcherPackageNames = null; - private DateTime? _lastLauncherSetBuilt = null; - private TimeSpan _rebuildLauncherSpan = TimeSpan.FromHours(1); - - public AutofillService() - { - _appSettings = Resolver.Resolve(); - } - - public override void OnAccessibilityEvent(AccessibilityEvent e) - { - try - { - var powerManager = (PowerManager)GetSystemService(PowerService); - if(Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch && !powerManager.IsInteractive) - { - return; - } - else if(Build.VERSION.SdkInt < BuildVersionCodes.Lollipop && !powerManager.IsScreenOn) - { - return; - } - - if(SkipPackage(e?.PackageName)) - { - return; - } - - var root = RootInActiveWindow; - if(root == null || root.PackageName != e.PackageName) - { - return; - } - - // var testNodes = GetWindowNodes(root, e, n => n.ViewIdResourceName != null && n.Text != null, false); - // var testNodesData = testNodes.Select(n => new { id = n.ViewIdResourceName, text = n.Text }); - - var notificationManager = (NotificationManager)GetSystemService(NotificationService); - var cancelNotification = true; - - switch(e.EventType) - { - case EventTypes.ViewFocused: - if(e.Source == null || !e.Source.Password || !_appSettings.AutofillPasswordField) - { - break; - } - - if(e.PackageName == BitwardenPackage) - { - CancelNotification(notificationManager); - break; - } - - if(ScanAndAutofill(root, e, notificationManager, cancelNotification)) - { - CancelNotification(notificationManager); - } - break; - case EventTypes.WindowContentChanged: - case EventTypes.WindowStateChanged: - if(_appSettings.AutofillPasswordField && e.Source.Password) - { - break; - } - else if(_appSettings.AutofillPasswordField && AutofillActivity.LastCredentials == null) - { - if(string.IsNullOrWhiteSpace(_lastNotificationUri)) - { - CancelNotification(notificationManager); - break; - } - - var uri = GetUri(root); - if(uri != _lastNotificationUri) - { - CancelNotification(notificationManager); - } - else if(uri.StartsWith(App.Constants.AndroidAppProtocol)) - { - CancelNotification(notificationManager, 30000); - } - - break; - } - - if(e.PackageName == BitwardenPackage) - { - CancelNotification(notificationManager); - break; - } - - if(_appSettings.AutofillPersistNotification) - { - var uri = GetUri(root); - if(uri != null && !uri.Contains(BitwardenWebsite)) - { - var needToFill = NeedToAutofill(AutofillActivity.LastCredentials, uri); - if(needToFill) - { - var passwordNodes = GetWindowNodes(root, e, n => n.Password, false); - needToFill = passwordNodes.Any(); - if(needToFill) - { - var allEditTexts = GetWindowNodes(root, e, n => EditText(n), false); - var usernameEditText = allEditTexts.TakeWhile(n => !n.Password).LastOrDefault(); - FillCredentials(usernameEditText, passwordNodes); - - allEditTexts.Dispose(); - usernameEditText.Dispose(); - } - passwordNodes.Dispose(); - } - - if(!needToFill) - { - NotifyToAutofill(uri, notificationManager); - cancelNotification = false; - } - } - - AutofillActivity.LastCredentials = null; - } - else - { - cancelNotification = ScanAndAutofill(root, e, notificationManager, cancelNotification); - } - - if(cancelNotification) - { - CancelNotification(notificationManager); - } - break; - default: - break; - } - - notificationManager?.Dispose(); - root.Dispose(); - e.Dispose(); - } - // Suppress exceptions so that service doesn't crash - catch { } - } - - public override void OnInterrupt() - { - - } - - public bool ScanAndAutofill(AccessibilityNodeInfo root, AccessibilityEvent e, - NotificationManager notificationManager, bool cancelNotification) - { - var passwordNodes = GetWindowNodes(root, e, n => n.Password, false); - if(passwordNodes.Count > 0) - { - var uri = GetUri(root); - if(uri != null && !uri.Contains(BitwardenWebsite)) - { - if(NeedToAutofill(AutofillActivity.LastCredentials, uri)) - { - var allEditTexts = GetWindowNodes(root, e, n => EditText(n), false); - var usernameEditText = allEditTexts.TakeWhile(n => !n.Password).LastOrDefault(); - FillCredentials(usernameEditText, passwordNodes); - - allEditTexts.Dispose(); - usernameEditText.Dispose(); - } - else - { - NotifyToAutofill(uri, notificationManager); - cancelNotification = false; - } - } - - AutofillActivity.LastCredentials = null; - } - else if(AutofillActivity.LastCredentials != null) - { - System.Threading.Tasks.Task.Run(async () => - { - await System.Threading.Tasks.Task.Delay(1000); - AutofillActivity.LastCredentials = null; - }); - } - - passwordNodes.Dispose(); - return cancelNotification; - } - - public void CancelNotification(NotificationManager notificationManager, long limit = 250) - { - if(Java.Lang.JavaSystem.CurrentTimeMillis() - _lastNotificationTime < limit) - { - return; - } - - _lastNotificationUri = null; - notificationManager?.Cancel(AutoFillNotificationId); - } - - private string GetUri(AccessibilityNodeInfo root) - { - var uri = string.Concat(App.Constants.AndroidAppProtocol, root.PackageName); - if(SupportedBrowsers.ContainsKey(root.PackageName)) - { - var addressNode = root.FindAccessibilityNodeInfosByViewId( - $"{root.PackageName}:id/{SupportedBrowsers[root.PackageName].UriViewId}").FirstOrDefault(); - if(addressNode != null) - { - uri = ExtractUri(uri, addressNode, SupportedBrowsers[root.PackageName]); - addressNode.Dispose(); - } - } - - return uri; - } - - private string ExtractUri(string uri, AccessibilityNodeInfo addressNode, Browser browser) - { - if(addressNode?.Text != null) - { - uri = browser.GetUriFunction(addressNode.Text)?.Trim(); - if(uri != null && uri.Contains(".")) - { - if(!uri.Contains("://") && !uri.Contains(" ")) - { - uri = string.Concat("http://", uri); - } - else if(Build.VERSION.SdkInt <= BuildVersionCodes.KitkatWatch) - { - var parts = uri.Split(new string[] { ". " }, StringSplitOptions.None); - if(parts.Length > 1) - { - var urlPart = parts.FirstOrDefault(p => p.StartsWith("http")); - if(urlPart != null) - { - uri = urlPart.Trim(); - } - } - } - } - } - - return uri; - } - - /// - /// Check to make sure it is ok to autofill still on the current screen - /// - private bool NeedToAutofill(AutofillCredentials creds, string currentUriString) - { - if(creds == null) - { - return false; - } - - Uri lastUri, currentUri; - if(Uri.TryCreate(creds.LastUri, UriKind.Absolute, out lastUri) && - Uri.TryCreate(currentUriString, UriKind.Absolute, out currentUri) && - lastUri.Host == currentUri.Host) - { - return true; - } - - return false; - } - - private static bool EditText(AccessibilityNodeInfo n) - { - return n?.ClassName?.Contains("EditText") ?? false; - } - - private void NotifyToAutofill(string uri, NotificationManager notificationManager) - { - if(notificationManager == null || string.IsNullOrWhiteSpace(uri)) - { - return; - } - - var now = Java.Lang.JavaSystem.CurrentTimeMillis(); - var intent = new Intent(this, typeof(AutofillActivity)); - intent.PutExtra("uri", uri); - intent.SetFlags(ActivityFlags.NewTask | ActivityFlags.SingleTop | ActivityFlags.ClearTop); - var pendingIntent = PendingIntent.GetActivity(this, 0, intent, PendingIntentFlags.UpdateCurrent); - - var notificationContent = Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch ? - AppResources.BitwardenAutofillServiceNotificationContent : - AppResources.BitwardenAutofillServiceNotificationContentOld; - - var builder = new Notification.Builder(this); - builder.SetSmallIcon(Resource.Drawable.notification_sm) - .SetContentTitle(AppResources.BitwardenAutofillService) - .SetContentText(notificationContent) - .SetTicker(notificationContent) - .SetWhen(now) - .SetContentIntent(pendingIntent); - - if(Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch) - { - builder.SetVisibility(NotificationVisibility.Secret) - .SetColor(global::Android.Support.V4.Content.ContextCompat.GetColor(ApplicationContext, - Resource.Color.primary)); - } - - if(Build.VERSION.SdkInt >= BuildVersionCodes.O) - { - if(_notificationChannel == null) - { - _notificationChannel = new NotificationChannel("bitwarden_autofill_service", - AppResources.AutofillService, NotificationImportance.Low); - notificationManager.CreateNotificationChannel(_notificationChannel); - } - builder.SetChannelId(_notificationChannel.Id); - } - - if(/*Build.VERSION.SdkInt <= BuildVersionCodes.N && */_appSettings.AutofillPersistNotification) - { - builder.SetPriority(-2); - } - - _lastNotificationTime = now; - _lastNotificationUri = uri; - notificationManager.Notify(AutoFillNotificationId, builder.Build()); - - builder.Dispose(); - } - - private void FillCredentials(AccessibilityNodeInfo usernameNode, IEnumerable passwordNodes) - { - FillEditText(usernameNode, AutofillActivity.LastCredentials?.Username); - foreach(var n in passwordNodes) - { - FillEditText(n, AutofillActivity.LastCredentials?.Password); - } - } - - private static void FillEditText(AccessibilityNodeInfo editTextNode, string value) - { - if(editTextNode == null || value == null) - { - return; - } - - var bundle = new Bundle(); - bundle.PutString(AccessibilityNodeInfo.ActionArgumentSetTextCharsequence, value); - editTextNode.PerformAction(global::Android.Views.Accessibility.Action.SetText, bundle); - } - - private NodeList GetWindowNodes(AccessibilityNodeInfo n, AccessibilityEvent e, - Func condition, bool disposeIfUnused, NodeList nodes = null, - int recursionDepth = 0) - { - if(nodes == null) - { - nodes = new NodeList(); - } - - var dispose = disposeIfUnused; - if(n != null && recursionDepth < 50) - { - if(n.WindowId == e.WindowId && !(n.ViewIdResourceName?.StartsWith(SystemUiPackage) ?? false) && condition(n)) - { - dispose = false; - nodes.Add(n); - } - - for(var i = 0; i < n.ChildCount; i++) - { - var childNode = n.GetChild(i); - if(i > 100) - { - global::Android.Util.Log.Info(BitwardenTag, "Too many child iterations."); - break; - } - else if(childNode.GetHashCode() == n.GetHashCode()) - { - global::Android.Util.Log.Info(BitwardenTag, - "Child node is the same as parent for some reason."); - } - else - { - GetWindowNodes(childNode, e, condition, true, nodes, recursionDepth++); - } - } - } - - if(dispose) - { - n?.Dispose(); - } - - return nodes; - } - - private bool SkipPackage(string eventPackageName) - { - if(string.IsNullOrWhiteSpace(eventPackageName) || FilteredPackageNames.Contains(eventPackageName) - || eventPackageName.Contains("launcher")) - { - return true; - } - - if(_launcherPackageNames == null || _lastLauncherSetBuilt == null || - (DateTime.Now - _lastLauncherSetBuilt.Value) > _rebuildLauncherSpan) - { - // refresh launcher list every now and then - _lastLauncherSetBuilt = DateTime.Now; - var intent = new Intent(Intent.ActionMain); - intent.AddCategory(Intent.CategoryHome); - var resolveInfo = PackageManager.QueryIntentActivities(intent, 0); - _launcherPackageNames = resolveInfo.Select(ri => ri.ActivityInfo.PackageName).ToHashSet(); - } - - return _launcherPackageNames.Contains(eventPackageName); - } - - public class Browser - { - public Browser(string packageName, string uriViewId) - { - PackageName = packageName; - UriViewId = uriViewId; - } - - public Browser(string packageName, string uriViewId, Func getUriFunction) - : this(packageName, uriViewId) - { - GetUriFunction = getUriFunction; - } - - public string PackageName { get; set; } - public string UriViewId { get; set; } - public Func GetUriFunction { get; set; } = (s) => s; - } - - public class NodeList : List, IDisposable - { - public void Dispose() - { - foreach(var item in this) - { - item.Dispose(); - } - } - } - } -}