问题描述
我提出一个简单的请求,只是检查我是否可以访问互联网:
return try {
val request = Request.Builder()
.url("https://www.google.com")
.build()
val response = OkHttpClient()
.newCall(request)
.execute()
response.body()?.close()
response.isSuccessful
} catch (e: IOException) {
false
}
通常我会和catch
一起陷入java.net.SocketTimeoutException: Read timed out
的圈套中。我发现OkHttp在okhttp3.internal.connection.RealConnection
的这一行中走得更远:
private void connectTls(ConnectionSpecSelector connectionSpecSelector) throws IOException {
// ...
try {
// ...
sslSocket.startHandshake(); // never goes to the next line
// ...
} catch(AssertionError e) {
// never goes here
} finally {
// goes here
}
}
我不知道在sslSocket.startHandshake()
中会发生什么,因为显然sslSocket
是com.android.org.conscrypt.ConscryptEngineSocket
,而Android并不想让我看看它是如何工作的。
是某种错误还是我做错了什么?是否有修复程序或至少有解决方法?
日志:
Accessing hidden method Lcom/android/org/conscrypt/ConscryptEngineSocket;->setUseSessionTickets(Z)V (blacklist,core-platform-api,reflection,denied)
Accessing hidden method Lcom/android/org/conscrypt/ConscryptEngineSocket;->setHostname(Ljava/lang/String;)V (blacklist,denied)
Accessing hidden method Lcom/android/org/conscrypt/ConscryptEngineSocket;->setUseSessionTickets(Z)V (blacklist,denied)
我使用Android 11(API 30)和OkHttp3 v3.12.0。
P.S。我记得我无法将OkHttp更新为较新的版本,因为我还有另一个依赖于此特定版本的OkHttp的依赖项,因为较新的版本向后不兼容。
解决方法
好像您的okhttp3库版本通过反射访问任何内部受限制的类(通过新的android版本)。
这里介绍了如何遵循以及如何为您找到最大的合适解决方案:https://developer.android.com/distribute/best-practices/develop/restrictions-non-sdk-interfaces
或...
一种快速的解决方法是将“ Android构建工具”版本降低为尚未实现此类限制的版本。是一个小技巧,可以避免对新的android版本的限制。但是要小心,因为playmarket不会接受使用过旧的“构建工具”编译的应用程序,因此它不是持久的解决方案。
,升级到OkHttp3 v3.12.12,此问题已得到解决。
请参见https://square.github.io/okhttp/changelog_3x/#version-3129