bitwarden-android/src/Android/AutofillService.cs

460 lines
18 KiB
C#
Raw Normal View History

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;
2017-05-29 18:38:03 +03:00
using Bit.App.Abstractions;
using XLabs.Ioc;
namespace Bit.Android
{
[Service(Permission = global::Android.Manifest.Permission.BindAccessibilityService, Label = "bitwarden")]
2017-01-24 07:32:52 +03:00
[IntentFilter(new string[] { "android.accessibilityservice.AccessibilityService" })]
[MetaData("android.accessibilityservice", Resource = "@xml/accessibilityservice")]
public class AutofillService : AccessibilityService
{
2017-01-24 07:32:52 +03:00
private const int AutoFillNotificationId = 34573;
private const string SystemUiPackage = "com.android.systemui";
2017-01-31 03:26:39 +03:00
private const string BitwardenPackage = "com.x8bit.bitwarden";
2017-02-01 04:45:51 +03:00
private const string BitwardenWebsite = "bitwarden.com";
private static Dictionary<string, Browser> SupportedBrowsers => new List<Browser>
{
new Browser("com.android.chrome", "url_bar"),
new Browser("com.chrome.beta", "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.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"),
2017-05-29 15:35:36 +03:00
new Browser("org.codeaurora.swe.browser", "url_bar"),
new Browser("org.iron.srware", "url_bar"),
2017-07-23 06:31:38 +03:00
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(' ').FirstOrDefault()),
new Browser("org.mozilla.firefox", "url_bar_title"),
new Browser("org.mozilla.firefox_beta", "url_bar_title"),
new Browser("org.mozilla.focus", "display_url"),
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"),
2017-02-17 07:09:40 +03:00
new Browser("com.linkbubble.playstore", "url_text"),
new Browser("com.ksmobile.cb", "address_bar_edit_text"),
new Browser("acr.browser.lightning", "search"),
2017-10-13 05:53:41 +03:00
new Browser("acr.browser.barebones", "search"),
new Browser("com.microsoft.emmx", "url_bar")
}.ToDictionary(n => n.PackageName);
2017-02-01 08:38:35 +03:00
2017-05-29 18:38:03 +03:00
private readonly IAppSettingsService _appSettings;
2017-05-30 15:18:56 +03:00
private long _lastNotificationTime = 0;
private string _lastNotificationUri = null;
2017-05-29 18:38:03 +03:00
public AutofillService()
{
_appSettings = Resolver.Resolve<IAppSettingsService>();
}
public override void OnAccessibilityEvent(AccessibilityEvent e)
{
var powerManager = (PowerManager)GetSystemService(PowerService);
2017-09-08 16:16:21 +03:00
if(Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch && !powerManager.IsInteractive)
{
return;
}
else if(Build.VERSION.SdkInt < BuildVersionCodes.Lollipop && !powerManager.IsScreenOn)
{
return;
}
try
2017-01-24 07:32:52 +03:00
{
var root = RootInActiveWindow;
if(e == null || root == null || string.IsNullOrWhiteSpace(e.PackageName) ||
e.PackageName == SystemUiPackage || e.PackageName.Contains("launcher") ||
root.PackageName != e.PackageName)
{
return;
}
2017-02-17 07:09:40 +03:00
2017-07-31 19:34:19 +03:00
//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 });
//testNodes.Dispose();
2017-02-09 02:19:59 +03:00
var notificationManager = (NotificationManager)GetSystemService(NotificationService);
2017-05-29 18:38:03 +03:00
var cancelNotification = true;
switch(e.EventType)
{
2017-05-29 18:38:03 +03:00
case EventTypes.ViewFocused:
if(!e.Source.Password || !_appSettings.AutofillPasswordField)
{
break;
}
2017-02-03 07:36:40 +03:00
if(e.PackageName == BitwardenPackage)
{
2017-05-29 18:38:03 +03:00
CancelNotification(notificationManager);
break;
}
2017-02-20 01:29:00 +03:00
2017-05-29 18:38:03 +03:00
if(ScanAndAutofill(root, e, notificationManager, cancelNotification))
{
2017-05-29 18:38:03 +03:00
CancelNotification(notificationManager);
}
break;
case EventTypes.WindowContentChanged:
case EventTypes.WindowStateChanged:
if(_appSettings.AutofillPasswordField && e.Source.Password)
{
break;
}
else if(_appSettings.AutofillPasswordField && AutofillActivity.LastCredentials == null)
2017-05-29 18:38:03 +03:00
{
2017-05-30 15:18:56 +03:00
if(string.IsNullOrWhiteSpace(_lastNotificationUri))
{
CancelNotification(notificationManager);
break;
}
var uri = GetUri(root);
if(uri != _lastNotificationUri)
{
CancelNotification(notificationManager);
}
2017-05-30 15:35:57 +03:00
else if(uri.StartsWith(App.Constants.AndroidAppProtocol))
{
CancelNotification(notificationManager, 30000);
2017-05-30 15:35:57 +03:00
}
break;
}
2017-05-29 18:38:03 +03:00
if(e.PackageName == BitwardenPackage)
{
CancelNotification(notificationManager);
break;
}
2017-05-29 18:38:03 +03:00
if(_appSettings.AutofillPersistNotification)
{
var uri = GetUri(root);
if(uri != null && !uri.Contains(BitwardenWebsite))
{
2017-05-29 18:38:03 +03:00
var needToFill = NeedToAutofill(AutofillActivity.LastCredentials, uri);
if(needToFill)
{
2017-05-29 18:38:03 +03:00
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();
}
2017-05-30 15:18:56 +03:00
passwordNodes.Dispose();
}
2017-05-29 18:38:03 +03:00
if(!needToFill)
{
NotifyToAutofill(uri, notificationManager);
cancelNotification = false;
}
}
AutofillActivity.LastCredentials = null;
}
2017-05-29 18:38:03 +03:00
else
{
cancelNotification = ScanAndAutofill(root, e, notificationManager, cancelNotification);
}
2017-01-24 07:32:52 +03:00
if(cancelNotification)
{
2017-05-29 18:38:03 +03:00
CancelNotification(notificationManager);
}
break;
default:
break;
}
2017-02-21 02:22:24 +03:00
notificationManager?.Dispose();
root.Dispose();
e.Dispose();
}
2017-05-20 18:29:20 +03:00
// Suppress exceptions so that service doesn't crash
catch { }
}
public override void OnInterrupt()
{
}
2017-05-29 18:38:03 +03:00
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;
});
2017-05-29 18:38:03 +03:00
}
passwordNodes.Dispose();
return cancelNotification;
}
2017-05-30 15:35:57 +03:00
public void CancelNotification(NotificationManager notificationManager, long limit = 250)
2017-05-29 18:38:03 +03:00
{
2017-05-30 15:35:57 +03:00
if(Java.Lang.JavaSystem.CurrentTimeMillis() - _lastNotificationTime < limit)
2017-05-29 18:38:03 +03:00
{
return;
}
2017-05-30 15:18:56 +03:00
_lastNotificationUri = null;
2017-05-29 18:38:03 +03:00
notificationManager?.Cancel(AutoFillNotificationId);
}
2017-02-01 04:45:51 +03:00
private string GetUri(AccessibilityNodeInfo root)
{
var uri = string.Concat(App.Constants.AndroidAppProtocol, root.PackageName);
if(SupportedBrowsers.ContainsKey(root.PackageName))
2017-02-01 04:45:51 +03:00
{
var addressNode = root.FindAccessibilityNodeInfosByViewId(
$"{root.PackageName}:id/{SupportedBrowsers[root.PackageName].UriViewId}").FirstOrDefault();
if(addressNode != null)
{
uri = ExtractUri(uri, addressNode, SupportedBrowsers[root.PackageName]);
2017-02-21 02:22:24 +03:00
addressNode.Dispose();
}
2017-02-01 04:45:51 +03:00
}
return uri;
}
private string ExtractUri(string uri, AccessibilityNodeInfo addressNode, Browser browser)
{
if(addressNode?.Text != null)
{
2017-02-18 05:18:59 +03:00
uri = browser.GetUriFunction(addressNode.Text).Trim();
if(uri != null && uri.Contains("."))
2017-02-01 06:53:32 +03:00
{
2017-02-18 05:18:59 +03:00
if(!uri.Contains("://") && !uri.Contains(" "))
2017-02-01 06:53:32 +03:00
{
uri = string.Concat("http://", uri);
}
else if(Build.VERSION.SdkInt <= BuildVersionCodes.KitkatWatch)
{
var parts = uri.Split(new string[] { ". " }, StringSplitOptions.None);
if(parts.Length > 1)
2017-02-01 06:53:32 +03:00
{
var urlPart = parts.FirstOrDefault(p => p.StartsWith("http"));
if(urlPart != null)
{
uri = urlPart.Trim();
}
2017-02-01 06:53:32 +03:00
}
}
}
}
2017-01-24 07:32:52 +03:00
return uri;
}
2017-01-31 03:26:39 +03:00
/// <summary>
/// Check to make sure it is ok to autofill still on the current screen
/// </summary>
private bool NeedToAutofill(AutofillCredentials creds, string currentUriString)
{
2017-01-31 03:26:39 +03:00
if(creds == null)
{
return false;
}
Uri lastUri, currentUri;
if(Uri.TryCreate(creds.LastUri, UriKind.Absolute, out lastUri) &&
2017-01-31 03:26:39 +03:00
Uri.TryCreate(currentUriString, UriKind.Absolute, out currentUri) &&
lastUri.Host == currentUri.Host)
2017-01-24 07:32:52 +03:00
{
return true;
}
return false;
}
2017-01-24 07:32:52 +03:00
private static bool EditText(AccessibilityNodeInfo n)
{
return n?.ClassName?.Contains("EditText") ?? false;
}
2017-02-21 02:22:24 +03:00
private void NotifyToAutofill(string uri, NotificationManager notificationManager)
{
if(notificationManager == null || string.IsNullOrWhiteSpace(uri))
{
return;
}
2017-05-29 18:38:03 +03:00
var now = Java.Lang.JavaSystem.CurrentTimeMillis();
2017-01-24 07:32:52 +03:00
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);
2017-09-08 16:16:21 +03:00
var notificationContent = Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch ?
App.Resources.AppResources.BitwardenAutofillServiceNotificationContent :
App.Resources.AppResources.BitwardenAutofillServiceNotificationContentOld;
var builder = new Notification.Builder(this);
2017-01-29 07:58:26 +03:00
builder.SetSmallIcon(Resource.Drawable.notification_sm)
2017-02-01 08:38:35 +03:00
.SetContentTitle(App.Resources.AppResources.BitwardenAutofillService)
2017-09-08 16:16:21 +03:00
.SetContentText(notificationContent)
.SetTicker(notificationContent)
2017-05-29 18:38:03 +03:00
.SetWhen(now)
2017-01-24 07:32:52 +03:00
.SetContentIntent(pendingIntent);
2017-01-29 07:58:26 +03:00
2017-02-01 04:45:51 +03:00
if(Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch)
{
builder.SetVisibility(NotificationVisibility.Secret)
.SetColor(global::Android.Support.V4.Content.ContextCompat.GetColor(ApplicationContext,
Resource.Color.primary));
}
2017-05-30 04:52:12 +03:00
if(/*Build.VERSION.SdkInt <= BuildVersionCodes.N && */_appSettings.AutofillPersistNotification)
2017-05-29 15:35:36 +03:00
{
builder.SetPriority(-2);
2017-05-29 15:35:36 +03:00
}
2017-05-30 15:18:56 +03:00
_lastNotificationTime = now;
_lastNotificationUri = uri;
2017-01-24 07:32:52 +03:00
notificationManager.Notify(AutoFillNotificationId, builder.Build());
2017-02-21 02:22:24 +03:00
builder.Dispose();
}
2017-01-24 07:32:52 +03:00
private void FillCredentials(AccessibilityNodeInfo usernameNode, IEnumerable<AccessibilityNodeInfo> passwordNodes)
{
FillEditText(usernameNode, AutofillActivity.LastCredentials?.Username);
2017-01-31 03:26:39 +03:00
foreach(var n in passwordNodes)
2017-01-24 07:32:52 +03:00
{
FillEditText(n, AutofillActivity.LastCredentials?.Password);
2017-01-24 07:32:52 +03:00
}
}
2017-01-24 07:32:52 +03:00
private static void FillEditText(AccessibilityNodeInfo editTextNode, string value)
{
2017-01-31 03:26:39 +03:00
if(editTextNode == null || value == null)
{
return;
}
2017-01-24 07:32:52 +03:00
var bundle = new Bundle();
bundle.PutString(AccessibilityNodeInfo.ActionArgumentSetTextCharsequence, value);
editTextNode.PerformAction(global::Android.Views.Accessibility.Action.SetText, bundle);
}
2017-02-21 02:22:24 +03:00
private NodeList GetWindowNodes(AccessibilityNodeInfo n, AccessibilityEvent e,
Func<AccessibilityNodeInfo, bool> condition, bool disposeIfUnused, NodeList nodes = null)
{
2017-02-18 18:50:27 +03:00
if(nodes == null)
{
2017-02-21 02:22:24 +03:00
nodes = new NodeList();
2017-02-18 18:50:27 +03:00
}
if(n != null)
{
2017-02-21 02:22:24 +03:00
var dispose = disposeIfUnused;
2017-02-17 06:22:19 +03:00
if(n.WindowId == e.WindowId && !(n.ViewIdResourceName?.StartsWith(SystemUiPackage) ?? false) && condition(n))
{
2017-02-21 02:22:24 +03:00
dispose = false;
2017-02-18 18:50:27 +03:00
nodes.Add(n);
}
2017-02-18 23:48:24 +03:00
for(var i = 0; i < n.ChildCount; i++)
{
2017-02-21 02:22:24 +03:00
GetWindowNodes(n.GetChild(i), e, condition, true, nodes);
}
if(dispose)
{
n.Dispose();
}
}
2017-02-18 18:50:27 +03:00
return nodes;
}
public class Browser
{
public Browser(string packageName, string uriViewId)
{
PackageName = packageName;
UriViewId = uriViewId;
}
public Browser(string packageName, string uriViewId, Func<string, string> getUriFunction)
: this(packageName, uriViewId)
{
GetUriFunction = getUriFunction;
}
public string PackageName { get; set; }
public string UriViewId { get; set; }
public Func<string, string> GetUriFunction { get; set; } = (s) => s;
}
2017-02-21 02:22:24 +03:00
public class NodeList : List<AccessibilityNodeInfo>, IDisposable
{
public void Dispose()
{
foreach(var item in this)
{
item.Dispose();
}
}
}
}
2017-01-28 07:32:48 +03:00
}