DownloadFtp.cpp
上传用户:oadesign
上传日期:2013-12-25
资源大小:265k
文件大小:9k
源码类别:

进程与线程

开发平台:

Visual C++

  1. // DownloadFtp.cpp: implementation of the CDownloadFtp class.
  2. //
  3. //////////////////////////////////////////////////////////////////////
  4. #include "stdafx.h"
  5. #include "NetDownMTR.h"
  6. #include "DownloadFtp.h"
  7. #ifdef _DEBUG
  8. #undef THIS_FILE
  9. static char THIS_FILE[]=__FILE__;
  10. #define new DEBUG_NEW
  11. #endif
  12. void DownloadNotify ( int nIndex, UINT nNotityType, LPVOID lpNotifyData, LPVOID pDownloadMTR );
  13. //////////////////////////////////////////////////////////////////////
  14. // Construction/Destruction
  15. //////////////////////////////////////////////////////////////////////
  16. CDownloadFtp::CDownloadFtp()
  17. {
  18. }
  19. CDownloadFtp::~CDownloadFtp()
  20. {
  21. }
  22. //
  23. // 创建一个 FTP 的数据通道连接
  24. //
  25. BOOL CDownloadFtp::CreateFTPDataConnect(CSocketClient &SocketClient)
  26. {
  27. ASSERT ( SocketClient.m_hSocket == INVALID_SOCKET );
  28. // 设被动模式
  29. if ( !m_SocketClient.SendString ( "PASVrn" ) ) return FALSE;
  30. CString csResponseStr;
  31. if ( !m_SocketClient.GetResponse ( 227, &csResponseStr ) )
  32. return FALSE;
  33. CString csPasvIP;
  34. USHORT uPasvPort = 0;
  35. if ( !m_SocketClient.GetIPAndPortByPasvString ( csResponseStr, csPasvIP, uPasvPort ) )
  36. return FALSE;
  37. // 创建数据通道连接
  38. if ( !SocketClient.Connect ( csPasvIP, uPasvPort ) )
  39. return FALSE;
  40. return TRUE;
  41. }
  42. //
  43. // 连接需要下载的服务器
  44. //
  45. BOOL CDownloadFtp::Connect ()
  46. {
  47. if ( !CDownloadPub::Connect () )
  48. return FALSE;
  49. int nResponseCode = m_SocketClient.GetResponse ();
  50. if ( nResponseCode != 220 )
  51. {
  52. Log ( L_WARNING, "Connect to [%s:%u] failed", m_csServer, m_nPort );
  53. }
  54. // 登录
  55. if ( m_csUsername.IsEmpty () )
  56. m_csUsername = "anonymous";
  57. if ( !m_SocketClient.SendString ( "USER %srn", m_csUsername ) ) return FALSE;
  58. nResponseCode = m_SocketClient.GetResponse ();
  59. // 需要密码
  60. if ( nResponseCode == 331 )
  61. {
  62. if ( !m_SocketClient.SendString ( "PASS %srn", m_csPassword ) ) return FALSE;
  63. nResponseCode = m_SocketClient.GetResponse ();
  64. }
  65. // 登录失败
  66. if ( nResponseCode != 230 )
  67. {
  68. Log ( L_ERROR, "Login failed" );
  69. return FALSE;
  70. }
  71. return TRUE;
  72. }
  73. BOOL CDownloadFtp::DownloadOnce()
  74. {
  75. // 不需要下载了
  76. int nWillDownloadSize = Get_WillDownloadSize(); // 本次应该下载的字节数
  77. int nDownloadedSize = Get_DownloadedSize (); // 已下载字节数
  78. if ( nWillDownloadSize > 0 && nDownloadedSize >= nWillDownloadSize )
  79. return DownloadEnd(TRUE);
  80. if ( !CDownloadPub::DownloadOnce () )
  81. return DownloadEnd(FALSE);
  82. // 设置接收数据类型为 二进制模式
  83. if ( !m_SocketClient.SendString ( "TYPE Irn" ) )
  84. return DownloadEnd(FALSE);
  85. if ( !m_SocketClient.GetResponse ( 200 ) )
  86. return DownloadEnd(FALSE);
  87. SLEEP_RETURN_Down ( 0 );
  88. // 创建数据通道连接
  89. CSocketClient SocketClient;
  90. if ( !CreateFTPDataConnect ( SocketClient ) )
  91. return DownloadEnd(FALSE);
  92. SLEEP_RETURN_Down ( 0 );
  93. // 设定下载文件的起始位置
  94. int nWillDownloadStartPos = Get_WillDownloadStartPos (); // 开始位置
  95. if ( !m_SocketClient.SendString ( "REST %drn", nWillDownloadStartPos+nDownloadedSize ) )
  96. return DownloadEnd(FALSE);
  97. if ( !m_SocketClient.GetResponse ( 350 ) )
  98. return DownloadEnd(FALSE);
  99. // 提交下载文件的请求
  100. if ( !m_SocketClient.SendString ( "RETR %srn", m_csObject ) )
  101. return DownloadEnd(FALSE);
  102. if ( !m_SocketClient.GetResponse ( 150 ) )
  103. return DownloadEnd(FALSE);
  104. SLEEP_RETURN_Down ( 0 );
  105. // 从数据通道读取数据,并保存到文件中
  106. BOOL bRet = RecvDataAndSaveToFile(SocketClient);
  107. SocketClient.Disconnect ();
  108. return DownloadEnd ( bRet );
  109. }
  110. //
  111. // 获取远程站点信息,如:是否支持断点续传、要下载的文件大小和创建时间等
  112. //
  113. BOOL CDownloadFtp::GetRemoteSiteInfo()
  114. {
  115. BOOL bRet = FALSE;
  116. CSocketClient SocketClient;
  117. CString csReadData;
  118. char szRecvBuf[NET_BUFFER_SIZE] = {0};
  119. int nReadSize = 0;
  120. CString csOneLine;
  121. int i;
  122. if ( !CDownloadPub::GetRemoteSiteInfo() )
  123. goto Finished;
  124. if ( !m_SocketClient.Is_Connected () && !Connect () )
  125. goto Finished;
  126. // 设置接收数据类型为 ASCII
  127. if ( !m_SocketClient.SendString ( "TYPE Arn" ) ) goto Finished;
  128. if ( !m_SocketClient.GetResponse ( 200 ) )
  129. goto Finished;
  130. SLEEP_RETURN ( 0 );
  131. // 判断是否支持断点续传
  132. if ( !m_SocketClient.SendString ( "REST 1rn" ) ) goto Finished;
  133. m_bSupportResume = ( m_SocketClient.GetResponse () == 350 );
  134. TRACE ( "是否支持断点续传:%sn", m_bSupportResume ? "支持":"不支持" );
  135. if ( m_bSupportResume )
  136. {
  137. if ( !m_SocketClient.SendString ( "REST 0rn" ) ) goto Finished;
  138. VERIFY ( m_SocketClient.GetResponse ( 350 ) );
  139. }
  140. SLEEP_RETURN ( 0 );
  141. // 创建数据通道连接
  142. if ( !CreateFTPDataConnect ( SocketClient ) )
  143. goto Finished;
  144. // 发送列举文件命令
  145. if ( !m_SocketClient.SendString ( "LIST %srn", m_csObject ) )
  146. goto Finished;
  147. if ( !m_SocketClient.GetResponse ( 150 ) )
  148. goto Finished;
  149. SLEEP_RETURN ( 0 );
  150. // 从数据通道读取文件列表信息,直到命令通道返回 "226" 回应码,注意:这里接收数据用非阻塞型的。
  151. for ( i=0; ;i++ )
  152. {
  153. memset ( szRecvBuf, 0, sizeof(szRecvBuf) );
  154. nReadSize = SocketClient.Receive ( szRecvBuf, sizeof(szRecvBuf), FALSE );
  155. if ( nReadSize < 0 )
  156. {
  157. Log ( L_WARNING, "Receive file list info failed" );
  158. break;
  159. }
  160. csReadData += szRecvBuf;
  161. int nResponseCode = m_SocketClient.GetResponse ( (CString*)NULL, FALSE );
  162. if ( nResponseCode == -1 ) goto Finished;
  163. else if ( nResponseCode == 0 )
  164. {
  165. SLEEP_RETURN ( 100 );
  166. }
  167. else if ( nResponseCode == 226 )
  168. {
  169. break;
  170. }
  171. else goto Finished;
  172. SLEEP_RETURN ( 0 );
  173. }
  174. for ( i=0; !csReadData.IsEmpty() ; i++ )
  175. {
  176. csOneLine = GetOneLine ( csReadData );
  177. if ( !csOneLine.IsEmpty() )
  178. {
  179. ParseFileInfoStr ( csOneLine );
  180. }
  181. }
  182. bRet = TRUE;
  183. Finished:
  184. m_SocketClient.Disconnect ();
  185. return bRet;
  186. }
  187. //
  188. // 从 "-rw-rw-rw-   1 user     group    37979686 Mar  9 13:39 FTP-Callgle 1.4.0_20060309.cab" 字符串中分析
  189. // 出文件信息
  190. //
  191. void CDownloadFtp::ParseFileInfoStr(CString &csFileInfoStr)
  192. {
  193. csFileInfoStr.TrimLeft (); csFileInfoStr.TrimRight ();
  194. if ( csFileInfoStr.IsEmpty() ) return;
  195. BOOL bLastCharIsSpace = ( csFileInfoStr[0]==' ' );
  196. int nSpaceCount = 0;
  197. CString csFileTime1, csFileTime2, csFileTime3;
  198. CString csNodeStr;
  199. CString csFileName;
  200. for ( int i=0; i<csFileInfoStr.GetLength(); i++ )
  201. {
  202. if ( csFileInfoStr[i]==' ' )
  203. {
  204. if ( !bLastCharIsSpace )
  205. {
  206. csNodeStr = csFileInfoStr.Mid ( i );
  207. csNodeStr.TrimLeft (); csNodeStr.TrimRight ();
  208. nSpaceCount ++;
  209. int nFindPos = csNodeStr.Find ( " ", 0 );
  210. if ( nFindPos < 0 ) nFindPos = csNodeStr.GetLength() - 1;
  211. csNodeStr = csNodeStr.Left ( nFindPos );
  212. // 文件大小
  213. if ( nSpaceCount == 4 )
  214. {
  215. if ( m_nIndex == -1 ) // 主线程才需要获取文件大小的信息
  216. {
  217. m_nFileTotalSize = (int)atoi(csNodeStr);
  218. DownloadNotify ( m_nIndex, NOTIFY_TYPE_GOT_REMOTE_FILESIZE, (LPVOID)m_nFileTotalSize, m_pDownloadMTR );
  219. int nWillDownloadStartPos = Get_WillDownloadStartPos (); // 开始位置
  220. int nWillDownloadSize = Get_WillDownloadSize(); // 本次应该下载的字节数
  221. int nDownloadedSize = Get_DownloadedSize (); // 已下载字节数
  222. if ( m_nFileTotalSize > 0 && nWillDownloadSize-nDownloadedSize > m_nFileTotalSize )
  223. Set_WillDownloadSize ( m_nFileTotalSize-nDownloadedSize );
  224. }
  225. }
  226. // 文件时间第一节
  227. else if ( nSpaceCount == 5 )
  228. {
  229. csFileTime1 = csNodeStr;
  230. }
  231. // 文件时间第二节
  232. else if ( nSpaceCount == 6 )
  233. {
  234. csFileTime2 = csNodeStr;
  235. }
  236. // 文件时间第三节
  237. else if ( nSpaceCount == 7 )
  238. {
  239. csFileTime3 = csNodeStr;
  240. }
  241. else if ( nSpaceCount > 7 )
  242. {
  243. csFileName = csFileInfoStr.Mid ( i );
  244. csFileName.TrimLeft(); csFileName.TrimRight();
  245. break;
  246. }
  247. }
  248. bLastCharIsSpace = TRUE;
  249. }
  250. else
  251. {
  252. bLastCharIsSpace = FALSE;
  253. }
  254. }
  255. GetFileTimeInfo ( csFileTime1, csFileTime2, csFileTime3 );
  256. }
  257. void CDownloadFtp::GetFileTimeInfo(CString csFileTime1, CString csFileTime2, CString csFileTime3)
  258. {
  259. if ( csFileTime3.IsEmpty() ) return;
  260. CString csYear, csMonth, csDay, csHour, csMinute, csSecond;
  261. int nMonth = GetMouthByShortStr ( csFileTime1 );
  262. ASSERT ( nMonth >= 1 && nMonth <= 12 );
  263. csMonth.Format ( "%02d", nMonth );
  264. COleDateTime tOleNow = COleDateTime::GetCurrentTime ();
  265. csDay.Format ( "%02d", atoi(csFileTime2) );
  266. csSecond = "00";
  267. // 带年的日期,如:Aug 21 2006
  268. if ( csFileTime3.Find ( ":", 0 ) < 0 )
  269. {
  270. csYear = csFileTime3;
  271. csHour = "00";
  272. csMinute = "00";
  273. }
  274. // 不带年的日期,如:Feb 21 01:11
  275. else
  276. {
  277. csYear.Format ( "%04d", tOleNow.GetYear() );
  278. int nPos = csFileTime3.Find ( ":", 0 );
  279. if ( nPos < 0 ) nPos = csFileTime3.GetLength() - 1;
  280. csHour = csFileTime3.Left ( nPos );
  281. csMinute = csFileTime3.Mid ( nPos+1 );
  282. }
  283. CString csFileTimeInfo;
  284. csFileTimeInfo.Format ( "%s-%s-%s %s:%s:%s", csYear, csMonth, csDay, csHour, csMinute, csSecond );
  285. ConvertStrToCTime ( csFileTimeInfo.GetBuffer(0), m_TimeLastModified );
  286. }