📄 openssladapter.cc
字号:
return SOCKET_ERROR; case SSL_CONNECTED: break; case SSL_ERROR: default: return SOCKET_ERROR; } // OpenSSL will return an error if we try to write zero bytes if (cb == 0) return 0; ssl_write_needs_read_ = false; int code = SSL_write(ssl_, pv, cb); switch (SSL_get_error(ssl_, code)) { case SSL_ERROR_NONE: //LOG(LS_INFO) << " -- success"; return code; case SSL_ERROR_WANT_READ: //LOG(LS_INFO) << " -- error want read"; ssl_write_needs_read_ = true; SetError(EWOULDBLOCK); break; case SSL_ERROR_WANT_WRITE: //LOG(LS_INFO) << " -- error want write"; SetError(EWOULDBLOCK); break; case SSL_ERROR_ZERO_RETURN: //LOG(LS_INFO) << " -- remote side closed"; SetError(EWOULDBLOCK); // do we need to signal closure? break; default: //LOG(LS_INFO) << " -- error " << code; Error("SSL_write", (code ? code : -1), false); break; } return SOCKET_ERROR;}intOpenSSLAdapter::Recv(void* pv, size_t cb) { //LOG(LS_INFO) << "OpenSSLAdapter::Recv(" << cb << ")"; switch (state_) { case SSL_NONE: return AsyncSocketAdapter::Recv(pv, cb); case SSL_WAIT: case SSL_CONNECTING: SetError(EWOULDBLOCK); return SOCKET_ERROR; case SSL_CONNECTED: break; case SSL_ERROR: default: return SOCKET_ERROR; } // Don't trust OpenSSL with zero byte reads if (cb == 0) return 0; ssl_read_needs_write_ = false; int code = SSL_read(ssl_, pv, cb); switch (SSL_get_error(ssl_, code)) { case SSL_ERROR_NONE: //LOG(LS_INFO) << " -- success"; return code; case SSL_ERROR_WANT_READ: //LOG(LS_INFO) << " -- error want read"; SetError(EWOULDBLOCK); break; case SSL_ERROR_WANT_WRITE: //LOG(LS_INFO) << " -- error want write"; ssl_read_needs_write_ = true; SetError(EWOULDBLOCK); break; case SSL_ERROR_ZERO_RETURN: //LOG(LS_INFO) << " -- remote side closed"; SetError(EWOULDBLOCK); // do we need to signal closure? break; default: //LOG(LS_INFO) << " -- error " << code; Error("SSL_read", (code ? code : -1), false); break; } return SOCKET_ERROR;}intOpenSSLAdapter::Close() { Cleanup(); state_ = restartable_ ? SSL_WAIT : SSL_NONE; return AsyncSocketAdapter::Close();}Socket::ConnStateOpenSSLAdapter::GetState() const { //if (signal_close_) // return CS_CONNECTED; ConnState state = socket_->GetState(); if ((state == CS_CONNECTED) && ((state_ == SSL_WAIT) || (state_ == SSL_CONNECTING))) state = CS_CONNECTING; return state;}voidOpenSSLAdapter::OnConnectEvent(AsyncSocket* socket) { LOG(LS_INFO) << "OpenSSLAdapter::OnConnectEvent"; if (state_ != SSL_WAIT) { ASSERT(state_ == SSL_NONE); AsyncSocketAdapter::OnConnectEvent(socket); return; } state_ = SSL_CONNECTING; if (int err = BeginSSL()) { AsyncSocketAdapter::OnCloseEvent(socket, err); }}voidOpenSSLAdapter::OnReadEvent(AsyncSocket* socket) { //LOG(LS_INFO) << "OpenSSLAdapter::OnReadEvent"; if (state_ == SSL_NONE) { AsyncSocketAdapter::OnReadEvent(socket); return; } if (state_ == SSL_CONNECTING) { if (int err = ContinueSSL()) { Error("ContinueSSL", err); } return; } if (state_ != SSL_CONNECTED) return; // Don't let ourselves go away during the callbacks //PRefPtr<OpenSSLAdapter> lock(this); // TODO: fix this if (ssl_write_needs_read_) { //LOG(LS_INFO) << " -- onStreamWriteable"; AsyncSocketAdapter::OnWriteEvent(socket); } //LOG(LS_INFO) << " -- onStreamReadable"; AsyncSocketAdapter::OnReadEvent(socket);}voidOpenSSLAdapter::OnWriteEvent(AsyncSocket* socket) { //LOG(LS_INFO) << "OpenSSLAdapter::OnWriteEvent"; if (state_ == SSL_NONE) { AsyncSocketAdapter::OnWriteEvent(socket); return; } if (state_ == SSL_CONNECTING) { if (int err = ContinueSSL()) { Error("ContinueSSL", err); } return; } if (state_ != SSL_CONNECTED) return; // Don't let ourselves go away during the callbacks //PRefPtr<OpenSSLAdapter> lock(this); // TODO: fix this if (ssl_read_needs_write_) { //LOG(LS_INFO) << " -- onStreamReadable"; AsyncSocketAdapter::OnReadEvent(socket); } //LOG(LS_INFO) << " -- onStreamWriteable"; AsyncSocketAdapter::OnWriteEvent(socket);}voidOpenSSLAdapter::OnCloseEvent(AsyncSocket* socket, int err) { LOG(LS_INFO) << "OpenSSLAdapter::OnCloseEvent(" << err << ")"; AsyncSocketAdapter::OnCloseEvent(socket, err);}// This code is taken from the "Network Security with OpenSSL"// sample in chapter 5boolOpenSSLAdapter::SSLPostConnectionCheck(SSL* ssl, const char* host) { if (!host) return false; // Checking the return from SSL_get_peer_certificate here is not strictly // necessary. With our setup, it is not possible for it to return // NULL. However, it is good form to check the return. X509* certificate = SSL_get_peer_certificate(ssl); if (!certificate) return false;#ifdef _DEBUG { LOG(LS_INFO) << "Certificate from server:"; BIO* mem = BIO_new(BIO_s_mem()); X509_print_ex(mem, certificate, XN_FLAG_SEP_CPLUS_SPC, X509_FLAG_NO_HEADER); BIO_write(mem, "\0", 1); char* buffer; BIO_get_mem_data(mem, &buffer); LOG(LS_INFO) << buffer; BIO_free(mem); char* cipher_description = SSL_CIPHER_description(SSL_get_current_cipher(ssl), NULL, 128); LOG(LS_INFO) << "Cipher: " << cipher_description; OPENSSL_free(cipher_description); }#endif bool ok = false; int extension_count = X509_get_ext_count(certificate); for (int i = 0; i < extension_count; ++i) { X509_EXTENSION* extension = X509_get_ext(certificate, i); int extension_nid = OBJ_obj2nid(X509_EXTENSION_get_object(extension)); if (extension_nid == NID_subject_alt_name) { X509V3_EXT_METHOD* meth = X509V3_EXT_get(extension); if (!meth) break; void* ext_str = NULL;#if OPENSSL_VERSION_NUMBER >= 0x0090800fL const unsigned char **ext_value_data = (const_cast<const unsigned char **> (&extension->value->data));#else unsigned char **ext_value_data = &extension->value->data;#endif if (meth->it) { ext_str = ASN1_item_d2i(NULL, ext_value_data, extension->value->length, ASN1_ITEM_ptr(meth->it)); } else { ext_str = meth->d2i(NULL, ext_value_data, extension->value->length); } STACK_OF(CONF_VALUE)* value = meth->i2v(meth, ext_str, NULL); for (int j = 0; j < sk_CONF_VALUE_num(value); ++j) { CONF_VALUE* nval = sk_CONF_VALUE_value(value, j); if (!strcmp(nval->name, "DNS") && !strcmp(nval->value, host)) { ok = true; break; } } } if (ok) break; } char data[256]; X509_name_st* subject; if (!ok && (subject = X509_get_subject_name(certificate)) && (X509_NAME_get_text_by_NID(subject, NID_commonName, data, sizeof(data)) > 0)) { data[sizeof(data)-1] = 0; if (_stricmp(data, host) == 0) ok = true; } X509_free(certificate); if (!ok && ignore_bad_cert()) { LOG(LS_WARNING) << "TLS certificate check FAILED. " << "Allowing connection anyway."; ok = true; } if (ok) ok = (SSL_get_verify_result(ssl) == X509_V_OK); if (!ok && ignore_bad_cert()) { LOG(LS_INFO) << "Other TLS post connection checks failed."; ok = true; } return ok;}#if _DEBUG// We only use this for tracing and so it is only needed in debug modevoidOpenSSLAdapter::SSLInfoCallback(const SSL* s, int where, int ret) { const char* str = "undefined"; int w = where & ~SSL_ST_MASK; if (w & SSL_ST_CONNECT) { str = "SSL_connect"; } else if (w & SSL_ST_ACCEPT) { str = "SSL_accept"; } if (where & SSL_CB_LOOP) { LOG(LS_INFO) << str << ":" << SSL_state_string_long(s); } else if (where & SSL_CB_ALERT) { str = (where & SSL_CB_READ) ? "read" : "write"; LOG(LS_INFO) << "SSL3 alert " << str << ":" << SSL_alert_type_string_long(ret) << ":" << SSL_alert_desc_string_long(ret); } else if (where & SSL_CB_EXIT) { if (ret == 0) { LOG(LS_INFO) << str << ":failed in " << SSL_state_string_long(s); } else if (ret < 0) { LOG(LS_INFO) << str << ":error in " << SSL_state_string_long(s); } }}#endif // _DEBUGintOpenSSLAdapter::SSLVerifyCallback(int ok, X509_STORE_CTX* store) {#if _DEBUG if (!ok) { char data[256]; X509* cert = X509_STORE_CTX_get_current_cert(store); int depth = X509_STORE_CTX_get_error_depth(store); int err = X509_STORE_CTX_get_error(store); LOG(LS_INFO) << "Error with certificate at depth: " << depth; X509_NAME_oneline(X509_get_issuer_name(cert), data, sizeof(data)); LOG(LS_INFO) << " issuer = " << data; X509_NAME_oneline(X509_get_subject_name(cert), data, sizeof(data)); LOG(LS_INFO) << " subject = " << data; LOG(LS_INFO) << " err = " << err << ":" << X509_verify_cert_error_string(err); }#endif // Get our stream pointer from the store SSL* ssl = reinterpret_cast<SSL*>( X509_STORE_CTX_get_ex_data(store, SSL_get_ex_data_X509_STORE_CTX_idx())); OpenSSLAdapter* stream = reinterpret_cast<OpenSSLAdapter*>(SSL_get_app_data(ssl)); if (!ok && stream->ignore_bad_cert()) { LOG(LS_WARNING) << "Ignoring cert error while verifying cert chain"; ok = 1; } return ok;}SSL_CTX*OpenSSLAdapter::SetupSSLContext() { SSL_CTX* ctx = SSL_CTX_new(TLSv1_client_method()); if (ctx == NULL) return NULL; // Add the root cert to the SSL context#if OPENSSL_VERSION_NUMBER >= 0x0090800fL const unsigned char* cert_buffer#else unsigned char* cert_buffer#endif = EquifaxSecureGlobalEBusinessCA1_certificate; size_t cert_buffer_len = sizeof(EquifaxSecureGlobalEBusinessCA1_certificate); X509* cert = d2i_X509(NULL, &cert_buffer, cert_buffer_len); if (cert == NULL) { SSL_CTX_free(ctx); return NULL; } if (!X509_STORE_add_cert(SSL_CTX_get_cert_store(ctx), cert)) { X509_free(cert); SSL_CTX_free(ctx); return NULL; }#ifdef _DEBUG SSL_CTX_set_info_callback(ctx, SSLInfoCallback);#endif SSL_CTX_set_verify(ctx, SSL_VERIFY_PEER, SSLVerifyCallback); SSL_CTX_set_verify_depth(ctx, 4); SSL_CTX_set_cipher_list(ctx, "ALL:!ADH:!LOW:!EXP:!MD5:@STRENGTH"); return ctx;}} // namespace talk_base#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -