⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 openssladapter.cc

📁 本人收集整理的一份c/c++跨平台网络库
💻 CC
📖 第 1 页 / 共 2 页
字号:
    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 + -