FtpGetFile WinINEt永远不会回来

时间:2013-09-12 16:40:56

标签: c++ ftp hang wininet

我遇到了一个奇怪的问题(非常奇怪,让我说嘿嘿)。在FTP下载EXE文件(24 MB)期间,如果连接中断,则看起来Wininet库的函数FtpGetFile有一个错误,它永远不会返回。这会导致将来的文件传输失败(连接已打开)。 显然,我通过增加服务器传输的超时找到了一种解决方法,但我不喜欢它。我没有通过谷歌搜索找到类似的问题(也许我介绍了错误的关键字)。

我在互联网上阅读了一些论坛,似乎每个人都不建议使用FtpGetFile,因为它是错误的。

这出现在一个网络场景中,该场景有很大的延迟(并非总是如此),但在良好的条件下它会消失(下载正确进行并且FtpGetFile始终返回)。

以下是我使用该功能的方法:

if( FtpGetFile(m_hFtpSession, strSourcePath.c_str(), strTargetPath.c_str(), 0, 0, FTP_TRANSFER_TYPE_BINARY, 0)==TRUE)

任何人都可以确认吗?我应该重构我的代码并寻找更新吗?

谢谢

1 个答案:

答案 0 :(得分:0)

我找到了一种不使用FtpGetFile下载文件的方法。我希望这段代码可以帮助别人:

bool RetrieveFile(const string& strSource, const string& strTarget) {

   /* The handle for the transfer */
   HINTERNET hTransfer = NULL;

   /*
    * Set default error
    */
   DWORD error = ERROR_SUCCESS;

   if( !isConnected ) {
     debug("%s(): ERROR not connected\n", __FUNCTION__);
     return false;
   }

   /* Initiate access to a remote FTP connection */
   hTransfer = FtpOpenFile(hFtpSession, strSource.c_str(), GENERIC_READ,
       FTP_TRANSFER_TYPE_BINARY, 0);

   if(hTransfer) {
      std::ofstream myostream(strTarget.c_str(), std::ios::binary);
      if ( myostream.is_open() ) {
         static const DWORD SIZE = 1024;

         BYTE data[SIZE];
         DWORD size = 0;
         do {
            BOOL result = InternetReadFile(hTransfer, data, SIZE, &size);
            if ( result == FALSE ) {
               error = GetLastError();
               Debug("InternetReadFile(): %lu\n", error);
            }
            myostream.write((const char*)data, size);
         }
         while ((error == ERROR_SUCCESS) && (size > 0));

         // Close the stream
         myostream.close();
      }
      else {
         Debug("Could not open '%s'.\n", strTarget.c_str());
         error = ERROR_FILE_NOT_FOUND; // Not necessarily not found, but it is to describe a file error which is different from ERROR_SUCCESS
      }

      // Close
      const BOOL result = InternetCloseHandle(hTransfer);
      if ( result == FALSE ) {
         const DWORD error = GetLastError();
         debug("InternetClose(): %lu\n", error);
      }

      /* Check error status of the process */
      return (error == ERROR_SUCCESS);
   }

   DWORD dwInetError;
   DWORD dwExtLength = 1000;
   TCHAR *szExtErrMsg = NULL;
   TCHAR errmsg[1000];
   szExtErrMsg = errmsg;
   int returned = InternetGetLastResponseInfo( &dwInetError, szExtErrMsg, &dwExtLength );
   debug("dwInetError: %d  Returned: %d\n", dwInetError, returned);
   debug("Buffer: %s\n", szExtErrMsg);

   debug("%s() : ERROR to get '%s' file (errorCode=%d)\n", __FUNCTION__, strSource.c_str(), GetLastError());

   return false;
}