From cad84002994cced9b65beaf74faed641ddad1f3b Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 10:27:29 -0400 Subject: [PATCH 1/6] Add support for PAC proxies Signed-off-by: Rafal Augustyniak --- .../engine/AndroidProxyMonitor.java | 24 +++++++++++++++---- 1 file changed, 20 insertions(+), 4 deletions(-) diff --git a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java index d912b4e5be..daf3051508 100644 --- a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java +++ b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java @@ -9,6 +9,7 @@ import android.net.Proxy; import android.net.ProxyInfo; import android.os.Build; +import android.os.Bundle; @TargetApi(Build.VERSION_CODES.LOLLIPOP) public class AndroidProxyMonitor extends BroadcastReceiver { @@ -34,7 +35,6 @@ private AndroidProxyMonitor(Context context, EnvoyEngine envoyEngine) { this.connectivityManager = (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE); registerReceiver(context); - this.handleProxyChange(); } private void registerReceiver(Context context) { @@ -45,15 +45,31 @@ private void registerReceiver(Context context) { @Override public void onReceive(Context context, Intent intent) { - handleProxyChange(); + handleProxyChange(intent); } - private void handleProxyChange() { - ProxyInfo info = connectivityManager.getDefaultProxy(); + private void handleProxyChange(final Intent intent) { + ProxyInfo info = this.extractProxyInfo(intent); + if (info == null) { envoyEngine.setProxySettings("", 0); } else { envoyEngine.setProxySettings(info.getHost(), info.getPort()); } } + + private ProxyInfo extractProxyInfo(final Intent intent) { + ProxyInfo info = connectivityManager.getDefaultProxy(); + + if (info.getPacFileUrl() != null) { + Bundle extras = intent.getExtras(); + if (extras == null) { + return null; + } + + info = (ProxyInfo)extras.get("android.intent.extra.PROXY_INFO"); + } + + return info; + } } From ac98a0a32e09ae70bdd14afe5b36d480409ddadd Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 14:45:45 -0400 Subject: [PATCH 2/6] Fix tests Signed-off-by: Rafal Augustyniak --- .../engine/AndroidProxyMonitor.java | 19 +++++++++++++++---- .../PerformHTTPRequestUsingProxyTest.kt | 18 +++++++++++------- .../PerformHTTPSRequestBadHostname.kt | 19 +++++++++++-------- .../PerformHTTPSRequestUsingAsyncProxyTest.kt | 17 ++++++++++------- .../PerformHTTPSRequestUsingProxyTest.kt | 19 +++++++++++-------- 5 files changed, 58 insertions(+), 34 deletions(-) diff --git a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java index daf3051508..9b50823176 100644 --- a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java +++ b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java @@ -8,16 +8,17 @@ import android.net.ConnectivityManager; import android.net.Proxy; import android.net.ProxyInfo; +import android.net.Uri; import android.os.Build; import android.os.Bundle; @TargetApi(Build.VERSION_CODES.LOLLIPOP) -public class AndroidProxyMonitor extends BroadcastReceiver { - private static volatile AndroidProxyMonitor instance = null; +class AndroidProxyMonitor extends BroadcastReceiver { + static volatile AndroidProxyMonitor instance = null; private ConnectivityManager connectivityManager; private EnvoyEngine envoyEngine; - public static void load(Context context, EnvoyEngine envoyEngine) { + static void load(Context context, EnvoyEngine envoyEngine) { if (instance != null) { return; } @@ -61,7 +62,17 @@ private void handleProxyChange(final Intent intent) { private ProxyInfo extractProxyInfo(final Intent intent) { ProxyInfo info = connectivityManager.getDefaultProxy(); - if (info.getPacFileUrl() != null) { + // If proxy is configured using the PAC file use the + // Android's injected localhost HTTP proxy. + // + // Android's injected localhost proxy can be accessed using a proxy host + // equal to `localhost` and a proxy port retrieved from intent's 'extras'. + // We cannot take a proxy port from the ProxyInfo object that's exposed by + // the connectivity manager as it's always equal to -1 for cases when PAC + // proxy is configured. + // + // See https://github.com/envoyproxy/envoy-mobile/issues/2531 for more details. + if (info.getPacFileUrl() != null && info.getPacFileUrl() != Uri.EMPTY) { Bundle extras = intent.getExtras(); if (extras == null) { return null; diff --git a/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt index e9fbe38ba6..ef29b4a499 100644 --- a/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt @@ -1,7 +1,10 @@ package test.kotlin.integration.proxying +import android.content.Intent import android.content.Context +import android.os.UserHandle import android.net.ConnectivityManager +import android.net.Proxy import android.net.ProxyInfo import androidx.test.core.app.ApplicationProvider @@ -47,17 +50,16 @@ class PerformHTTPRequestUsingProxy { fun `performs an HTTP request through a proxy`() { val port = (10001..11000).random() - val mockContext = Mockito.mock(Context::class.java) - Mockito.`when`(mockContext.getApplicationContext()).thenReturn(mockContext) - val mockConnectivityManager = Mockito.mock(ConnectivityManager::class.java) - Mockito.`when`(mockContext.getSystemService(Mockito.anyString())).thenReturn(mockConnectivityManager) - Mockito.`when`(mockConnectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) + val context = Mockito.spy(ApplicationProvider.getApplicationContext()) + val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) + Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) val onProxyEngineRunningLatch = CountDownLatch(1) val onEngineRunningLatch = CountDownLatch(1) val onRespondeHeadersLatch = CountDownLatch(1) - val proxyEngineBuilder = Proxy(ApplicationProvider.getApplicationContext(), port) + val proxyEngineBuilder = Proxy(context, port) .http() val proxyEngine = proxyEngineBuilder .addLogLevel(LogLevel.DEBUG) @@ -67,7 +69,9 @@ class PerformHTTPRequestUsingProxy { onProxyEngineRunningLatch.await(10, TimeUnit.SECONDS) assertThat(onProxyEngineRunningLatch.count).isEqualTo(0) - val builder = AndroidEngineBuilder(mockContext) + context.sendStickyBroadcast(Intent(Proxy.PROXY_CHANGE_ACTION)) + + val builder = AndroidEngineBuilder(context) val engine = builder .addLogLevel(LogLevel.DEBUG) .enableProxying(true) diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt index 61aa7637d5..68edfcc0c1 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt @@ -1,8 +1,10 @@ package test.kotlin.integration.proxying - +import android.content.Intent import android.content.Context +import android.os.UserHandle import android.net.ConnectivityManager +import android.net.Proxy import android.net.ProxyInfo import androidx.test.core.app.ApplicationProvider @@ -49,17 +51,16 @@ class PerformHTTPSRequestBadHostname { fun `attempts an HTTPs request through a proxy using an async DNS resolution that fails`() { val port = (10001..11000).random() - val mockContext = Mockito.mock(Context::class.java) - Mockito.`when`(mockContext.getApplicationContext()).thenReturn(mockContext) - val mockConnectivityManager = Mockito.mock(ConnectivityManager::class.java) - Mockito.`when`(mockContext.getSystemService(Mockito.anyString())).thenReturn(mockConnectivityManager) - Mockito.`when`(mockConnectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("loopback", port)) + val context = Mockito.spy(ApplicationProvider.getApplicationContext()) + val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) + Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) val onEngineRunningLatch = CountDownLatch(1) val onProxyEngineRunningLatch = CountDownLatch(1) val onErrorLatch = CountDownLatch(1) - val proxyEngineBuilder = Proxy(ApplicationProvider.getApplicationContext(), port).https() + val proxyEngineBuilder = Proxy(context, port).https() val proxyEngine = proxyEngineBuilder .addLogLevel(LogLevel.DEBUG) .addDNSQueryTimeoutSeconds(2) @@ -69,7 +70,9 @@ class PerformHTTPSRequestBadHostname { onProxyEngineRunningLatch.await(10, TimeUnit.SECONDS) assertThat(onProxyEngineRunningLatch.count).isEqualTo(0) - val builder = AndroidEngineBuilder(mockContext) + context.sendStickyBroadcast(Intent(Proxy.PROXY_CHANGE_ACTION)) + + val builder = AndroidEngineBuilder(context) val engine = builder .addLogLevel(LogLevel.DEBUG) .enableProxying(true) diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt index 4544eb87c3..ac074f3bf4 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt @@ -1,8 +1,10 @@ package test.kotlin.integration.proxying - +import android.content.Intent import android.content.Context +import android.os.UserHandle import android.net.ConnectivityManager +import android.net.Proxy import android.net.ProxyInfo import androidx.test.core.app.ApplicationProvider @@ -49,11 +51,10 @@ class PerformHTTPSRequestUsingAsyncProxyTest { fun `performs an HTTPs request through a proxy using async DNS resolution`() { val port = (10001..11000).random() - val mockContext = Mockito.mock(Context::class.java) - Mockito.`when`(mockContext.getApplicationContext()).thenReturn(mockContext) - val mockConnectivityManager = Mockito.mock(ConnectivityManager::class.java) - Mockito.`when`(mockContext.getSystemService(Mockito.anyString())).thenReturn(mockConnectivityManager) - Mockito.`when`(mockConnectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("localhost", port)) + val context = Mockito.spy(ApplicationProvider.getApplicationContext()) + val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) + Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) val onEngineRunningLatch = CountDownLatch(1) val onProxyEngineRunningLatch = CountDownLatch(1) @@ -68,7 +69,9 @@ class PerformHTTPSRequestUsingAsyncProxyTest { onProxyEngineRunningLatch.await(10, TimeUnit.SECONDS) assertThat(onProxyEngineRunningLatch.count).isEqualTo(0) - val builder = AndroidEngineBuilder(mockContext) + context.sendStickyBroadcast(Intent(Proxy.PROXY_CHANGE_ACTION)) + + val builder = AndroidEngineBuilder(context) val engine = builder .addLogLevel(LogLevel.DEBUG) .enableProxying(true) diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt index af260b5dfc..329d7898d0 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt @@ -1,8 +1,10 @@ package test.kotlin.integration.proxying - +import android.content.Intent import android.content.Context +import android.os.UserHandle import android.net.ConnectivityManager +import android.net.Proxy import android.net.ProxyInfo import androidx.test.core.app.ApplicationProvider @@ -49,17 +51,16 @@ class PerformHTTPSRequestUsingProxy { fun `performs an HTTPs request through a proxy`() { val port = (10001..11000).random() - val mockContext = Mockito.mock(Context::class.java) - Mockito.`when`(mockContext.getApplicationContext()).thenReturn(mockContext) - val mockConnectivityManager = Mockito.mock(ConnectivityManager::class.java) - Mockito.`when`(mockContext.getSystemService(Mockito.anyString())).thenReturn(mockConnectivityManager) - Mockito.`when`(mockConnectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) + val context = Mockito.spy(ApplicationProvider.getApplicationContext()) + val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) + Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) val onEngineRunningLatch = CountDownLatch(1) val onProxyEngineRunningLatch = CountDownLatch(1) val onRespondeHeadersLatch = CountDownLatch(1) - val proxyEngineBuilder = Proxy(ApplicationProvider.getApplicationContext(), port).https() + val proxyEngineBuilder = Proxy(context, port).https() val proxyEngine = proxyEngineBuilder .addLogLevel(LogLevel.DEBUG) .setOnEngineRunning { onProxyEngineRunningLatch.countDown() } @@ -68,7 +69,9 @@ class PerformHTTPSRequestUsingProxy { onProxyEngineRunningLatch.await(10, TimeUnit.SECONDS) assertThat(onProxyEngineRunningLatch.count).isEqualTo(0) - val builder = AndroidEngineBuilder(mockContext) + context.sendStickyBroadcast(Intent(Proxy.PROXY_CHANGE_ACTION)) + + val builder = AndroidEngineBuilder(context) val engine = builder .addLogLevel(LogLevel.DEBUG) .enableProxying(true) From 0f74facc8bc2b56d58f45e321acedb264e29b80e Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 14:46:47 -0400 Subject: [PATCH 3/6] remove import Signed-off-by: Rafal Augustyniak --- .../integration/proxying/PerformHTTPRequestUsingProxyTest.kt | 1 - .../integration/proxying/PerformHTTPSRequestBadHostname.kt | 1 - .../proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt | 1 - .../integration/proxying/PerformHTTPSRequestUsingProxyTest.kt | 1 - 4 files changed, 4 deletions(-) diff --git a/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt index ef29b4a499..b651c34f74 100644 --- a/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPRequestUsingProxyTest.kt @@ -2,7 +2,6 @@ package test.kotlin.integration.proxying import android.content.Intent import android.content.Context -import android.os.UserHandle import android.net.ConnectivityManager import android.net.Proxy import android.net.ProxyInfo diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt index 68edfcc0c1..350653cdf4 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt @@ -2,7 +2,6 @@ package test.kotlin.integration.proxying import android.content.Intent import android.content.Context -import android.os.UserHandle import android.net.ConnectivityManager import android.net.Proxy import android.net.ProxyInfo diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt index ac074f3bf4..b29e5ffadf 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt @@ -2,7 +2,6 @@ package test.kotlin.integration.proxying import android.content.Intent import android.content.Context -import android.os.UserHandle import android.net.ConnectivityManager import android.net.Proxy import android.net.ProxyInfo diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt index 329d7898d0..4ce45b6f04 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingProxyTest.kt @@ -2,7 +2,6 @@ package test.kotlin.integration.proxying import android.content.Intent import android.content.Context -import android.os.UserHandle import android.net.ConnectivityManager import android.net.Proxy import android.net.ProxyInfo From 93341e1cbe150a3562d8757260c8446447f2de8d Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 15:23:39 -0400 Subject: [PATCH 4/6] comment Signed-off-by: Rafal Augustyniak --- .../io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java index 9b50823176..741f8e53ab 100644 --- a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java +++ b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java @@ -62,7 +62,7 @@ private void handleProxyChange(final Intent intent) { private ProxyInfo extractProxyInfo(final Intent intent) { ProxyInfo info = connectivityManager.getDefaultProxy(); - // If proxy is configured using the PAC file use the + // If a proxy is configured using the PAC file use // Android's injected localhost HTTP proxy. // // Android's injected localhost proxy can be accessed using a proxy host From 69cdc80fe7e56640e06c3759cdde8392eb4c3566 Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 15:36:02 -0400 Subject: [PATCH 5/6] Code review Signed-off-by: Rafal Augustyniak --- .../io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java | 3 +++ 1 file changed, 3 insertions(+) diff --git a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java index 741f8e53ab..563194dd37 100644 --- a/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java +++ b/library/java/io/envoyproxy/envoymobile/engine/AndroidProxyMonitor.java @@ -61,6 +61,9 @@ private void handleProxyChange(final Intent intent) { private ProxyInfo extractProxyInfo(final Intent intent) { ProxyInfo info = connectivityManager.getDefaultProxy(); + if (info == null) { + return null; + } // If a proxy is configured using the PAC file use // Android's injected localhost HTTP proxy. From 41ae4804723cd5d345ceb5c22853762750fd14ba Mon Sep 17 00:00:00 2001 From: Rafal Augustyniak Date: Tue, 4 Oct 2022 15:49:57 -0400 Subject: [PATCH 6/6] revert unintended changes in tests Signed-off-by: Rafal Augustyniak --- .../integration/proxying/PerformHTTPSRequestBadHostname.kt | 2 +- .../proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt index 350653cdf4..2b55c04350 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestBadHostname.kt @@ -53,7 +53,7 @@ class PerformHTTPSRequestBadHostname { val context = Mockito.spy(ApplicationProvider.getApplicationContext()) val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) - Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("loopback", port)) val onEngineRunningLatch = CountDownLatch(1) val onProxyEngineRunningLatch = CountDownLatch(1) diff --git a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt index b29e5ffadf..7d88f91511 100644 --- a/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt +++ b/test/kotlin/integration/proxying/PerformHTTPSRequestUsingAsyncProxyTest.kt @@ -53,7 +53,7 @@ class PerformHTTPSRequestUsingAsyncProxyTest { val context = Mockito.spy(ApplicationProvider.getApplicationContext()) val connectivityManager: ConnectivityManager = Mockito.mock(ConnectivityManager::class.java) Mockito.doReturn(connectivityManager).`when`(context).getSystemService(Context.CONNECTIVITY_SERVICE) - Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("127.0.0.1", port)) + Mockito.`when`(connectivityManager.getDefaultProxy()).thenReturn(ProxyInfo.buildDirectProxy("localhost", port)) val onEngineRunningLatch = CountDownLatch(1) val onProxyEngineRunningLatch = CountDownLatch(1)