]> gitweb.stoutner.com Git - PrivacyBrowserAndroid.git/blobdiff - app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java
Add support for I2P and custom proxies. https://redmine.stoutner.com/issues/355
[PrivacyBrowserAndroid.git] / app / src / main / java / com / stoutner / privacybrowser / helpers / ProxyHelper.java
diff --git a/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java b/app/src/main/java/com/stoutner/privacybrowser/helpers/ProxyHelper.java
new file mode 100644 (file)
index 0000000..a666208
--- /dev/null
@@ -0,0 +1,163 @@
+/*
+ * Copyright © 2016-2019 Soren Stoutner <soren@stoutner.com>.
+ *
+ * This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
+ *
+ * Privacy Browser is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Privacy Browser is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Privacy Browser.  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.stoutner.privacybrowser.helpers;
+
+import android.annotation.SuppressLint;
+import android.content.Context;
+import android.content.Intent;
+import android.net.Proxy;
+import android.os.Build;
+import android.os.Parcelable;
+import android.util.ArrayMap;
+import android.util.Log;
+
+import com.stoutner.privacybrowser.activities.MainWebViewActivity;
+
+import java.lang.reflect.Field;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+
+public class ProxyHelper {
+    public static final String NONE = "None";
+    public static final String TOR = "Tor";
+    public static final String I2P = "I2P";
+    public static final String CUSTOM = "Custom";
+
+    public static void setProxy(Context context, String mode) {
+        // Initialize the proxy host and port strings.
+        String proxyHost = "0";
+        String proxyPort = "0";
+
+        // Run the commands that correlate to the proxy mode.
+        switch (mode) {
+            case NONE:
+                // Clear the proxy values.
+                System.clearProperty("proxyHost");
+                System.clearProperty("proxyHost");
+                break;
+
+            case TOR:
+                // Update the proxy host and port strings.
+                proxyHost = "localhost";
+                proxyPort = "8118";
+
+                // Set the proxy values
+                System.setProperty("proxyHost", proxyHost);
+                System.setProperty("proxyPort", proxyPort);
+
+                // Ask Orbot to connect if its current status is not `"ON"`.
+                if (!MainWebViewActivity.orbotStatus.equals("ON")) {
+                    // Request Orbot to start.
+                    Intent orbotIntent = new Intent("org.torproject.android.intent.action.START");
+
+                    // Send the intent to the Orbot package.
+                    orbotIntent.setPackage("org.torproject.android");
+
+                    // Request a status response be sent back to this package.
+                    orbotIntent.putExtra("org.torproject.android.intent.extra.PACKAGE_NAME", context.getPackageName());
+
+                    // Make it so.
+                    context.sendBroadcast(orbotIntent);
+                }
+                break;
+
+            case I2P:
+                // Update the proxy host and port strings.
+                proxyHost = "localhost";
+                proxyPort = "4444";
+
+                // Set the proxy values
+                System.setProperty("proxyHost", proxyHost);
+                System.setProperty("proxyPort", proxyPort);
+                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);
+                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);
+                            }
+
+                            // 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);
+        }
+    }
+}
\ No newline at end of file