无法使并发HTTPS请求与POCO

时间:2015-08-14 11:15:25

标签: c++ multithreading c++11 poco poco-libraries

我想使用Poco(1.6.0)同时向同一个端点发送一些https请求,但我不断地使用我所做的代码获得异常。 (但是,使用HTTP请求的相同代码工作正常......)

以下是我使用的代码:

Poco::Net::initializeSSL();

try {
   Poco::URI ep("https://stackoverflow.com/");

   Poco::SharedPtr<Poco::Net::InvalidCertificateHandler> ptrHandler
         = new Poco::Net::AcceptCertificateHandler(false);
   Poco::Net::Context::Ptr context = new Poco::Net::Context(Poco::Net::Context::CLIENT_USE,
                                                   "", "", "", Poco::Net::Context::VERIFY_STRICT,
                                                   9, true, "ALL:!ADH:!LOW:!EXP:!MD5:@STRENGTH");
   Poco::Net::SSLManager::instance().initializeClient(0, ptrHandler, context);

   for (int i = 0; i < 20; ++i)
   {
      // Start an asynchronous HTTPS request
      std::future<bool> f = std::async(std::launch::async, []() -> bool {
                  try {
                     Poco::URI ep("https://github.com/");
                      std::unique_ptr<Poco::Net::HTTPSClientSession> session(
                               new Poco::Net::HTTPSClientSession (ep.getHost(), ep.getPort()) );
                      Poco::Net::HTTPRequest request(Poco::Net::HTTPRequest::HTTP_GET,
                                             ep.getPathAndQuery() ,
                                             Poco::Net::HTTPMessage::HTTP_1_1);
                      request.write(std::cerr);
                      try {
                        session->sendRequest(request);
                      }
                      catch (const Poco::Exception& e) {
                         std::cerr << "Error A" << std::endl;
                         std::cerr << e.code() << " - " << e.displayText() << std::endl;
                         throw;
                      }

                      Poco::Net::HTTPResponse response;
                      try {
                        session->receiveResponse(response);
                      }
                      catch (const Poco::Exception& e) {
                         std::cerr << "Error B" << std::endl;
                         std::cerr << e.code() << " - " << e.displayText() << std::endl;
                         throw;
                      }
                      response.write(std::cerr);
                      std::this_thread::sleep_for(std::chrono::seconds(1));
                     std::cerr << "Async request ended" << std::endl;
                  }
                  catch (const Poco::Exception& e) {
                     std::cerr << "Error :-/" << std::endl;
                     std::cerr << e.code() << " - " << e.displayText() << std::endl;
                     throw;
                  }
                  std::cerr << "Async request ended 2" << std::endl;

                  return true;
      });

      // Generate several HTTPS requests until the async request is completed
      std::chrono::microseconds span (10);
      while (f.wait_for(span) != std::future_status::ready)
      {
         std::unique_ptr<Poco::Net::HTTPSClientSession> session(
                  new Poco::Net::HTTPSClientSession (ep.getHost(), ep.getPort()) );
         Poco::Net::HTTPRequest request(Poco::Net::HTTPRequest::HTTP_GET,
                                 ep.getPathAndQuery() ,
                                 Poco::Net::HTTPMessage::HTTP_1_1);
         request.write(std::cerr);
         try {
            session->sendRequest(request);
         }
         catch (const Poco::Exception& e) {

            std::cerr << "Error 1" << std::endl;
            std::cerr << e.code() << " - " << e.displayText() << std::endl;
            throw;
         }
         Poco::Net::HTTPResponse response;

         try {
            session->receiveResponse(response);
         }
         catch (const Poco::Exception& e) {
            std::cerr << "Error 2" << std::endl;
            std::cerr << e.code() << " - " << e.displayText() << std::endl;
            throw;
         }
         response.write(std::cerr);
      }
      f.get();
   }

}
catch (const Poco::Exception& e) {
   std::cerr << "ERROR :-(" << std::endl;
   std::cerr << e.code() << " - " << e.displayText() << std::endl;
}

Poco::Net::uninitializeSSL();

这是我遇到的问题:

当我在QtCreator调试器中运行代码时,我得到以下一个:异步请求结束后,主线程中的当前请求抛出Poco异常。

Error 1
4 - I/O error: Interrupted

但是,这似乎没有出现在QtCreator调试器之外。这有点奇怪。

这可能是一个并发问题,但我找不到我做错了什么,我在互联网上没有看到任何类似的问题。 你能帮我找一下我的错误吗?

谢谢。

编辑:简化了最小代码,并在问题说明中添加了一些细节。

EDIT2 :我似乎忘了提到最重要的信息,即我遇到问题的环境。它是以下:Ubuntu 14.04 LTS 64位,gcc 4.8.4,gdb 7.7.1,openssl 1.0.1f

1 个答案:

答案 0 :(得分:1)

问题来自太旧版本的GDB 。在 7.9版本中,GDB带有许多与多线程调试相关的改进,可用于调试此代码,没有任何问题。使用此代码,旧版本的GDB将失败