X-Git-Url: https://gitweb.stoutner.com/?p=PrivacyBrowserAndroid.git;a=blobdiff_plain;f=app%2Fsrc%2Fmain%2Fjava%2Fcom%2Fstoutner%2Fprivacybrowser%2Fhelpers%2FProxyHelper.java;h=dce61690937c639367b46aea0d267edf3f0b5180;hp=a666208668283620bec4d95e893c362fc559228a;hb=01d647c09b892384e0b42e86b000b5cb858f7a2b;hpb=81179d84ced6b43360d42a4b44eb8fb329532ff4 diff --git a/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java b/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java index a6662086..dce61690 100644 --- a/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java +++ b/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java @@ -22,17 +22,29 @@ package com.stoutner.privacybrowser.helpers; import android.annotation.SuppressLint; import android.content.Context; import android.content.Intent; +import android.content.SharedPreferences; import android.net.Proxy; +import android.net.Uri; import android.os.Build; import android.os.Parcelable; import android.util.ArrayMap; import android.util.Log; +import android.view.View; +import androidx.preference.PreferenceManager; +import androidx.webkit.ProxyConfig; +import androidx.webkit.ProxyController; +import androidx.webkit.WebViewFeature; + +import com.google.android.material.snackbar.Snackbar; + +import com.stoutner.privacybrowser.R; import com.stoutner.privacybrowser.activities.MainWebViewActivity; import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; +import java.util.concurrent.Executor; public class ProxyHelper { public static final String NONE = "None"; @@ -40,13 +52,16 @@ public class ProxyHelper { public static final String I2P = "I2P"; public static final String CUSTOM = "Custom"; - public static void setProxy(Context context, String mode) { + public static void setProxy(Context context, View activityView, String proxyMode) { // Initialize the proxy host and port strings. String proxyHost = "0"; String proxyPort = "0"; + // Create a proxy config builder. + ProxyConfig.Builder proxyConfigBuilder = new ProxyConfig.Builder(); + // Run the commands that correlate to the proxy mode. - switch (mode) { + switch (proxyMode) { case NONE: // Clear the proxy values. System.clearProperty("proxyHost"); @@ -54,17 +69,20 @@ public class ProxyHelper { break; case TOR: - // Update the proxy host and port strings. + // Update the proxy host and port strings. These can be removed once the minimum API >= 21. proxyHost = "localhost"; proxyPort = "8118"; - // Set the proxy values + // Set the proxy values. These can be removed once the minimum API >= 21. System.setProperty("proxyHost", proxyHost); System.setProperty("proxyPort", proxyPort); + // Add the proxy to the builder. The proxy config builder can use a SOCKS proxy. + proxyConfigBuilder.addProxyRule("socks://localhost:9050"); + // Ask Orbot to connect if its current status is not `"ON"`. if (!MainWebViewActivity.orbotStatus.equals("ON")) { - // Request Orbot to start. + // Create an intent to request Orbot to start. Intent orbotIntent = new Intent("org.torproject.android.intent.action.START"); // Send the intent to the Orbot package. @@ -79,85 +97,146 @@ public class ProxyHelper { break; case I2P: - // Update the proxy host and port strings. + // Update the proxy host and port strings. These can be removed once the minimum API >= 21. proxyHost = "localhost"; proxyPort = "4444"; - // Set the proxy values + // Set the proxy values. These can be removed once the minimum API >= 21. System.setProperty("proxyHost", proxyHost); System.setProperty("proxyPort", proxyPort); + + // Add the proxy to the builder. + proxyConfigBuilder.addProxyRule("http://localhost:4444"); break; case CUSTOM: - // Update the proxy host and port strings. - proxyHost = "0"; - proxyPort = "0"; - - // Set the proxy values - System.setProperty("proxyHost", proxyHost); - System.setProperty("proxyPort", proxyPort); + // Get a handle for the shared preferences. + SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context); + + // Get the custom proxy URL string. + String customProxyUrlString = sharedPreferences.getString("proxy_custom_url", context.getString(R.string.proxy_custom_url_default_value)); + + // Parse the custom proxy URL. + try { + // Convert the custom proxy URL string to a URI. + Uri customProxyUri = Uri.parse(customProxyUrlString); + + // Get the proxy host and port strings from the shared preferences. These can be removed once the minimum API >= 21. + proxyHost = customProxyUri.getHost(); + proxyPort = String.valueOf(customProxyUri.getPort()); + + // Set the proxy values. These can be removed once the minimum API >= 21. + System.setProperty("proxyHost", proxyHost); + System.setProperty("proxyPort", proxyPort); + + // Add the proxy to the builder. + proxyConfigBuilder.addProxyRule(customProxyUrlString); + } catch (Exception exception){ // The custom proxy URL is invalid. + // Display a Snackbar. + Snackbar.make(activityView, R.string.custom_proxy_invalid, Snackbar.LENGTH_LONG).show(); + } break; } - // Use reflection to apply the new proxy values. - try { - // Get the application and APK classes. Suppress the lint warning that reflection may not always work in the future and on all devices. - Class applicationClass = Class.forName("android.app.Application"); - @SuppressLint("PrivateApi") Class loadedApkClass = Class.forName("android.app.LoadedApk"); - - // Get the declared fields. Suppress the lint warning that `mLoadedApk` cannot be resolved. - @SuppressWarnings("JavaReflectionMemberAccess") Field methodLoadedApkField = applicationClass.getDeclaredField("mLoadedApk"); - Field methodReceiversField = loadedApkClass.getDeclaredField("mReceivers"); - - // Allow the values to be changed. - methodLoadedApkField.setAccessible(true); - methodReceiversField.setAccessible(true); - - // Get the APK object. - Object methodLoadedApkObject = methodLoadedApkField.get(context); - - // Get an array map of the receivers. - ArrayMap receivers = (ArrayMap) methodReceiversField.get(methodLoadedApkObject); - - // Set the proxy if the receivers has at least one entry. - if (receivers != null) { - for (Object receiverMap : receivers.values()) { - for (Object receiver : ((ArrayMap) receiverMap).keySet()) { - // Get the receiver class. - // `Class`, which is an `unbounded wildcard parameterized type`, must be used instead of `Class`, which is a `raw type`. Otherwise, `receiveClass.getDeclaredMethod()` is unhappy. - Class receiverClass = receiver.getClass(); - - // Apply the new proxy settings to any classes whose names contain `ProxyChangeListener`. - if (receiverClass.getName().contains("ProxyChangeListener")) { - // Get the `onReceive` method from the class. - Method onReceiveMethod = receiverClass.getDeclaredMethod("onReceive", Context.class, Intent.class); - - // Create a proxy change intent. - Intent proxyChangeIntent = new Intent(Proxy.PROXY_CHANGE_ACTION); - - if (Build.VERSION.SDK_INT >= 21) { - // Get a proxy info class. - // `Class`, which is an `unbounded wildcard parameterized type`, must be used instead of `Class`, which is a `raw type`. Otherwise, `proxyInfoClass.getMethod()` is unhappy. - Class proxyInfoClass = Class.forName("android.net.ProxyInfo"); - - // Get the build direct proxy method from the proxy info class. - Method buildDirectProxyMethod = proxyInfoClass.getMethod("buildDirectProxy", String.class, Integer.TYPE); - - // Populate a proxy info object with the new proxy information. - Object proxyInfoObject = buildDirectProxyMethod.invoke(proxyInfoClass, proxyHost, Integer.valueOf(proxyPort)); - - // Add the proxy info object into the proxy change intent. - proxyChangeIntent.putExtra("proxy", (Parcelable) proxyInfoObject); + // Apply the proxy settings + if (WebViewFeature.isFeatureSupported(WebViewFeature.PROXY_OVERRIDE)) { // The fancy new proxy config can be used because the API >= 21. + // Convert the proxy config builder into a proxy config. + ProxyConfig proxyConfig = proxyConfigBuilder.build(); + + // Get the proxy controller. + ProxyController proxyController = ProxyController.getInstance(); + + // Applying a proxy requires an executor. + Executor executor = runnable -> { + // Do nothing. + }; + + // Applying a proxy requires a runnable. + Runnable runnable = () -> { + // Do nothing. + }; + + // Apply the proxy settings. + if (proxyMode.equals(NONE)) { // Remove the proxy. + proxyController.clearProxyOverride(executor, runnable); + } else { // Apply the proxy. + try { + // Apply the proxy. + proxyController.setProxyOverride(proxyConfig, executor, runnable); + } catch (IllegalArgumentException exception) { // The proxy config is invalid. + // Display a Snackbar. + Snackbar.make(activityView, R.string.custom_proxy_invalid, Snackbar.LENGTH_LONG).show(); + } + } + } else { // The old proxy method must be used, either because an old WebView is installed or because the API == 19; + // Get a handle for the shared preferences. + SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context); + + // Check to make sure a SOCKS proxy is not selected. + if (proxyMode.equals(CUSTOM) && sharedPreferences.getString("proxy_custom_url", context.getString(R.string.proxy_custom_url_default_value)).startsWith("socks://")) { + // Display a Snackbar. + Snackbar.make(activityView, R.string.socks_proxies_do_not_work_on_kitkat, Snackbar.LENGTH_LONG).show(); + } else { // Use reflection to apply the new proxy values. + try { + // Get the application and APK classes. Suppress the lint warning that reflection may not always work in the future and on all devices. + Class applicationClass = Class.forName("android.app.Application"); + @SuppressLint("PrivateApi") Class loadedApkClass = Class.forName("android.app.LoadedApk"); + + // Get the declared fields. Suppress the lint warning that `mLoadedApk` cannot be resolved. + @SuppressWarnings("JavaReflectionMemberAccess") Field methodLoadedApkField = applicationClass.getDeclaredField("mLoadedApk"); + Field methodReceiversField = loadedApkClass.getDeclaredField("mReceivers"); + + // Allow the values to be changed. + methodLoadedApkField.setAccessible(true); + methodReceiversField.setAccessible(true); + + // Get the APK object. + Object methodLoadedApkObject = methodLoadedApkField.get(context); + + // Get an array map of the receivers. + ArrayMap receivers = (ArrayMap) methodReceiversField.get(methodLoadedApkObject); + + // Set the proxy if the receivers has at least one entry. + if (receivers != null) { + for (Object receiverMap : receivers.values()) { + for (Object receiver : ((ArrayMap) receiverMap).keySet()) { + // Get the receiver class. + // `Class`, which is an `unbounded wildcard parameterized type`, must be used instead of `Class`, which is a `raw type`. Otherwise, `receiveClass.getDeclaredMethod()` is unhappy. + Class receiverClass = receiver.getClass(); + + // Apply the new proxy settings to any classes whose names contain `ProxyChangeListener`. + if (receiverClass.getName().contains("ProxyChangeListener")) { + // Get the `onReceive` method from the class. + Method onReceiveMethod = receiverClass.getDeclaredMethod("onReceive", Context.class, Intent.class); + + // Create a proxy change intent. + Intent proxyChangeIntent = new Intent(Proxy.PROXY_CHANGE_ACTION); + + if (Build.VERSION.SDK_INT >= 21) { + // Get a proxy info class. + // `Class`, which is an `unbounded wildcard parameterized type`, must be used instead of `Class`, which is a `raw type`. Otherwise, `proxyInfoClass.getMethod()` is unhappy. + Class proxyInfoClass = Class.forName("android.net.ProxyInfo"); + + // Get the build direct proxy method from the proxy info class. + Method buildDirectProxyMethod = proxyInfoClass.getMethod("buildDirectProxy", String.class, Integer.TYPE); + + // Populate a proxy info object with the new proxy information. + Object proxyInfoObject = buildDirectProxyMethod.invoke(proxyInfoClass, proxyHost, Integer.valueOf(proxyPort)); + + // Add the proxy info object into the proxy change intent. + proxyChangeIntent.putExtra("proxy", (Parcelable) proxyInfoObject); + } + + // Pass the proxy change intent to the `onReceive` method of the receiver class. + onReceiveMethod.invoke(receiver, context, proxyChangeIntent); + } } - - // Pass the proxy change intent to the `onReceive` method of the receiver class. - onReceiveMethod.invoke(receiver, context, proxyChangeIntent); } } + } catch (ClassNotFoundException | NoSuchFieldException | IllegalAccessException | NoSuchMethodException | InvocationTargetException exception) { + Log.d("enableProxyThroughOrbot", "Exception: " + exception); } } - } catch (ClassNotFoundException | NoSuchFieldException | IllegalAccessException | NoSuchMethodException | InvocationTargetException exception) { - Log.d("enableProxyThroughOrbot", "Exception: " + exception); } } } \ No newline at end of file