android okhttp3 底层,Android okhttp3 SSL握手底层实现追踪

本文深入探讨了OkHttp3中SSL握手的实现过程,从tcp三次握手开始,通过Java代码分析`RealConnection#connectSocket`方法,然后进入`connectTls`方法调用`startHandshake`,并进一步查看`OpenSSLSocketImpl`的`startHandshake`的native方法`NativeCrypto.SSL_do_handshake`。在握手过程中,涉及到设置非阻塞模式、SSL_do_handshake的循环尝试、错误处理及选择器的使用,确保SSL连接的成功建立。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

对于https,在tcp三次握手后就会进行ssl的握手,ssl握手的详细过程网上介绍的不少。下面跟踪下okhttp3对于ssl握手的实现过程。java

在okhttp3.internal.io.RealConnection#connectSocket中git

private void connectSocket(int connectTimeout, int readTimeout, int writeTimeout,

ConnectionSpecSelector connectionSpecSelector) throws IOException {

rawSocket.setSoTimeout(readTimeout);

try {

Platform.get().connectSocket(rawSocket, route.socketAddress(), connectTimeout);

} catch (ConnectException e) {

throw new ConnectException("Failed to connect to " + route.socketAddress());

}

source = Okio.buffer(Okio.source(rawSocket));

sink = Okio.buffer(Okio.sink(rawSocket));

if (route.address().sslSocketFactory() != null) {

connectTls(readTimeout, writeTimeout, connectionSpecSelector);

} else {

protocol = Protocol.HTTP_1_1;

socket = rawSocket;

}

。。。

}

首先tcp三次握手:Platform.get().connectSocket

其次得到I/O流:source = Okio.buffer(Okio.source(rawSocket));sink = Okio.buffer(Okio.sink(rawSocket));

而后判断是否须要ssl,若是须要则进行ssl:connectTls(readTimeout, writeTimeout, connectionSpecSelector);github

在connectTls中,忽略其余的,ssl握手发生在web

sslSocket.startHandshake();

startHandshake的实如今org.conscrypt.OpenSSLSocketImpl#startHandshake中,

(/libcore/crypto/src/main/java/org/conscrypt/OpenSSLSocketImpl.java)app

@Override public synchronized void startHandshake() throws IOException {

。。。

int sslSessionNativePointer;

try {

sslSessionNativePointer = NativeCrypto.SSL_do_handshake(sslNativePointer,

socket.getFileDescriptor$(), this, getSoTimeout(), client, npnProtocols,client ? null : alpnProtocols);

} catch (CertificateException e) {

SSLHandshakeException wrapper = new SSLHandshakeException(e.getMessage());

wrapper.initCause(e);

throw wrapper;

}

。。。

}

调用的是native函数NativeCrypto.SSL_do_handshake(),实如今/libcore/crypto/src/main/native/org_conscrypt_NativeCrypto.cpp中,取出关键部分进行分析socket

static jlong NativeCrypto_SSL_do_handshake(JNIEnv* env, jclass, jlong ssl_address, jobject fdObject,jobject shc, jint timeout_millis, jboolean client_mode, jbyteArray npnProtocols,

jbyteArray alpnProtocols) {

。。。

/*

* Make socket non-blocking, so SSL_connect SSL_read() and SSL_write() don't hang

* forever and we can use select() to find out if the socket is ready.

*/

if (!setBlocking(fd.get(), false)) {

throwSSLExceptionStr(env, "Unable to make socket non blocking");

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake setBlocking => 0", ssl);

return 0;

}

。。。

ret = 0;

while (appData->aliveAndKicking) {

errno = 0;

if (!appData->setCallbackState(env, shc, fdObject, npnProtocols, alpnProtocols)) {

// SocketException thrown by NetFd.isClosed

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake setCallbackState => 0", ssl);

return 0;

}

ret = SSL_do_handshake(ssl);

appData->clearCallbackState();

// cert_verify_callback threw exception

if (env->ExceptionCheck()) {

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake exception => 0", ssl);

return 0;

}

// success case

if (ret == 1) {

break;

}

// retry case

if (errno == EINTR) {

continue;

}

// error case

int sslError = SSL_get_error(ssl, ret);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake ret=%d errno=%d sslError=%d timeout_millis=%d",

ssl, ret, errno, sslError, timeout_millis);

/*

* If SSL_do_handshake doesn't succeed due to the socket being

* either unreadable or unwritable, we use sslSelect to

* wait for it to become ready. If that doesn't happen

* before the specified timeout or an error occurs, we

* cancel the handshake. Otherwise we try the SSL_connect

* again.

*/

if (sslError == SSL_ERROR_WANT_READ || sslError == SSL_ERROR_WANT_WRITE) {

appData->waitingThreads++;

int selectResult = sslSelect(env, sslError, fdObject, appData, timeout_millis);

if (selectResult == THROWN_EXCEPTION) {

// SocketException thrown by NetFd.isClosed

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake sslSelect => 0", ssl);

return 0;

}

if (selectResult == -1) {

throwSSLExceptionWithSslErrors(env, ssl, SSL_ERROR_SYSCALL, "handshake error");

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake selectResult == -1 => 0", ssl);

return 0;

}

if (selectResult == 0) {

throwSocketTimeoutException(env, "SSL handshake timed out");

SSL_clear(ssl);

freeOpenSslErrorState();

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake selectResult == 0 => 0", ssl);

return 0;

}

} else {

// ALOGE("Unknown error %d during handshake", error);

break;

}

}

// clean error. See SSL_do_handshake(3SSL) man page.

if (ret == 0) {

/*

* The other side closed the socket before the handshake could be

* completed, but everything is within the bounds of the TLS protocol.

* We still might want to find out the real reason of the failure.

*/

int sslError = SSL_get_error(ssl, ret);

if (sslError == SSL_ERROR_NONE || (sslError == SSL_ERROR_SYSCALL && errno == 0)) {

throwSSLExceptionStr(env, "Connection closed by peer");

} else {

throwSSLExceptionWithSslErrors(env, ssl, sslError, "SSL handshake terminated");

}

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake clean error => 0", ssl);

return 0;

}

// unclean error. See SSL_do_handshake(3SSL) man page.

if (ret < 0) {

/*

* Translate the error and throw exception. We are sure it is an error

* at this point.

*/

int sslError = SSL_get_error(ssl, ret);

throwSSLExceptionWithSslErrors(env, ssl, sslError, "SSL handshake aborted");

SSL_clear(ssl);

JNI_TRACE("ssl=%p NativeCrypto_SSL_do_handshake unclean error => 0", ssl);

return 0;

}

。。。

}

(1)设置SSL为非阻塞模式

(2)在while里调用openssl的SSL_do_handshake进行握手,返回后检查是否握手成功了,成功则直接跳出while

(3)若是不成功,则检查是不是由于socket不能读写形成的

(4)若是是由于socket此时不能读写,则调用sellect进行等待,这次握手成功则返回,这次失败则继续while,即回到(2)

(5)若是不是由于socket读写的缘由,则握手失败,跳出while,进行失败缘由判断tcp

回答: 当使用conda创建的环境消失时,可以尝试以下解决方法。首先,可以使用命令"conda activate <env-path>"来激活该环境,这样当前环境会变成(env-path)。然后,使用命令"source activate"来确保conda默认的base环境已经改变。接下来,使用命令"conda deactivate"取消激活状态。最后,再次使用命令"conda activate <env-path>"重新激活想要指向的base环境,观察到terminal开头变成了(base),表示成功将base从miniconda换回到anaconda3。使用命令"conda env list"可以查看conda环境列表,确认环境是否恢复正常。\[2\]另外,如果想要安装虚拟环境中的所有依赖项,可以在已激活的虚拟环境中执行以下命令:"conda list --explicit > spec-file.txt"和"conda env update --prefix ./spec-file.txt --file spec-file.txt --prune"。这样可以将虚拟环境中的所有依赖项安装到指定的环境中。\[3\] #### 引用[.reference_title] - *1* [conda——虚拟环境消失](https://blog.youkuaiyun.com/Jialins_blog/article/details/124622152)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [conda环境名称消失问题](https://blog.youkuaiyun.com/wudiyouyou1994/article/details/118105221)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [关于python项目移动文件夹后conda创建的虚拟环境失效问题的解决方案](https://blog.youkuaiyun.com/qq_40837795/article/details/130528541)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值