android - SSL 握手中止 : ssl=0x74d2dc46c8: Failure in SSL library, 通常是协议(protocol)错误 OPENSSL_internal:SSLV3_ALERT_HANDSHAKE_FAILURE

标签 android retrofit retrofit2 okhttp

在改造中,我在请求中发送 XML,作为响应,服务器将发送 XML 作为响应。我正在使用代理来使用该 API。但我总是得到 - HTTP FAILED: java.io.IOException: unexpected end of stream on null。我已经尝试过 stackoverflow 中提到的解决方案,但没有任何效果。以下代码适用于改造客户 -

    fun getRetrofitClient(context: Context): RestAPI {

    if (mRetrofitClient == null) {

        val cf: CertificateFactory = CertificateFactory.getInstance("X.509")
        val cert: InputStream = context.getResources().openRawResource(R.raw.mycert)
        val ca: Certificate
        ca = try {
            cf.generateCertificate(cert)
        } finally {
            cert.close()
        }

         val keyStoreType: String = KeyStore.getDefaultType()
        val keyStore: KeyStore = KeyStore.getInstance(keyStoreType)
        keyStore.load(null, null)
        keyStore.setCertificateEntry("ca", ca)
        val proxy = Proxy(Proxy.Type.HTTP, InetSocketAddress("xxxx.xx.xx.xx",xxxx ))
        val tmfAlgorithm: String = TrustManagerFactory.getDefaultAlgorithm()
        val tmf: TrustManagerFactory = TrustManagerFactory.getInstance(tmfAlgorithm)
        tmf.init(keyStore)

        val sslContext: SSLContext = SSLContext.getInstance("TLSv1.2")
        sslContext.init(null, tmf.getTrustManagers(), null)

        val spec = ConnectionSpec.Builder(ConnectionSpec.MODERN_TLS)
            .tlsVersions(TlsVersion.TLS_1_2)
            .cipherSuites(
                CipherSuite.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256,
            CipherSuite.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256,
            CipherSuite.TLS_DHE_RSA_WITH_AES_128_GCM_SHA256,
            CipherSuite.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA,
            CipherSuite.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA,
            CipherSuite.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,
            CipherSuite.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,
            CipherSuite.TLS_ECDHE_ECDSA_WITH_RC4_128_SHA,
            CipherSuite.TLS_ECDHE_RSA_WITH_RC4_128_SHA,
            CipherSuite.TLS_DHE_RSA_WITH_AES_128_CBC_SHA,
            CipherSuite.TLS_DHE_DSS_WITH_AES_128_CBC_SHA,
            CipherSuite.TLS_DHE_RSA_WITH_AES_256_CBC_SHA
            )
            .build()


        val httpClient = OkHttpClient.Builder()
            .followRedirects(true)
        .followSslRedirects(true)
        .connectTimeout(10, TimeUnit.MINUTES)
        .readTimeout(10, TimeUnit.MINUTES)
        .writeTimeout(10, TimeUnit.MINUTES)
        .sslSocketFactory(socketFactory)
        .socketFactory(socketFactory)
        .hostnameVerifier(getHostnameVerifier())
        .retryOnConnectionFailure(true)
        .proxy(proxy)
        .connectionSpecs(Collections.singletonList(spec))



        val retrofit = Retrofit.Builder()
            .baseUrl("https://xxx.xx.xx.xx:xxxx/xx/")
            .addConverterFactory(SimpleXmlConverterFactory.create())
            .client(httpClient.build())
            .build()
        mRetrofitClient = retrofit.create(RestAPI::class.java!!)
    }
    return mRetrofitClient!!
}

fun getHostnameVerifier(): HostnameVerifier? {
    return object : HostnameVerifier {
        override fun verify(hostname: String?, session: SSLSession?): Boolean {
            return true
        }
    }
}

下面的代码是用于 Rest API -
interface RestAPI {
    @Headers("Content-Type: text/xml","Connection: keep-alive","Transfer-Encoding: chunked","Cache-Control: no-cache","Accept: */*","Host: xxx.xx.xx.xx:xxxx","Accept-Encoding: gzip, deflate, br")
    @POST("BillQuery/V1.0")
    abstract fun billQuery(@Body input:String): Call<String>
}

我也试过 Volley ,同样的问题。但它在 Postman 中运行良好。我添加了像 postman 这样的标题。还是不行。

请帮忙。

最佳答案

unexpected end of stream如果 Response-Headers 无法解析或者 Response-Header 中的长度 Content-Length 可以由 okhttp 抛出与响应正文的实际长度不匹配(参见 HTTP FAILED: java.io.IOException: unexpected end of stream exception while making https request )。

在您的情况下,okhttp 想要向 HTTP 代理发送未加密的连接,并且在此步骤中无法解析来自代理响应的响应头,我猜。代理给 okhttp 一个无效的答案。您可以尝试使用 postman 通过 HTTP 连接到代理吗?

关于android - SSL 握手中止 : ssl=0x74d2dc46c8: Failure in SSL library, 通常是协议(protocol)错误 OPENSSL_internal:SSLV3_ALERT_HANDSHAKE_FAILURE,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60862111/

相关文章:

android - Cordova/PhoneGap HTML5 地理围栏

android - Retrofit 中的错误 call.enqueue

android - 在 retrofit 和 rxjava 完成后调用一个函数

android - 改造 http 缓存控制检查发回缓存,即使它已过时并再次进行网络调用

android - 如果它们有嵌套数组,如何创建数据字段以在 retrofit2 后使用

java - 实现 AdMob 后出现问题

c# - 文件上传到 .net c# 服务器抛出 IOException

java - 改造 Https 调用给连接重置

android - 在 Android Studio 上连接到 GitLab 存储库

android - Retrofit + rxandroid 导致 kitkat 出现 stackoverflow 错误