QtFtp.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  1. #include "QtFtp.h"
  2. #include "LightLog.h"
  3. #include <QEventLoop>
  4. /* 构造函数 */
  5. QtFtp::QtFtp(QObject *parent)
  6. {
  7. /* 设置协议 */
  8. m_url.setScheme("ftp");
  9. m_manager.setParent(this);
  10. /* 设置超时定时器 */
  11. m_timer.setTimerType(Qt::PreciseTimer);
  12. m_timer.setSingleShot(true);
  13. connect(&m_timer,SIGNAL(timeout()),this,SLOT(do_timeout()));
  14. }
  15. QtFtp::~QtFtp()
  16. {
  17. if(m_file.isOpen())
  18. {
  19. m_file.close();
  20. }
  21. if(m_reply != nullptr)
  22. {
  23. m_reply->deleteLater();
  24. m_reply = nullptr;
  25. }
  26. }
  27. /* 设置目标主机IP和端口 */
  28. void QtFtp::setHostAndPort(const QString &host, int port)
  29. {
  30. m_url.setHost(host);
  31. m_url.setPort(port);
  32. }
  33. /* 设置目标设备的用户名和密码 */
  34. void QtFtp::setUserPasswd(const QString &user, const QString &pw)
  35. {
  36. m_url.setUserName(user);
  37. m_url.setPassword(pw);
  38. }
  39. /**
  40. * @brief 上传文件
  41. * @param file 要上传的文件名
  42. * @param path 目标计算机存放文件的路径
  43. */
  44. void QtFtp::putFile(const QString &fileName, const QString &farPath)
  45. {
  46. m_isFinished = false;
  47. m_result = false;
  48. QFile file(fileName);
  49. file.open(QFile::ReadOnly);
  50. QByteArray ba = file.readAll();
  51. file.close();
  52. m_url.setPath(farPath);
  53. m_reply = m_manager.put(QNetworkRequest(m_url),ba);
  54. /* 将信号进行一次转发 */
  55. connect(m_reply,SIGNAL(uploadProgress(qint64,qint64)),this,SIGNAL(signal_uploadProgress(qint64,qint64)));
  56. /* 连接信号和槽 */
  57. connect(m_reply,SIGNAL(finished()),this,SLOT(do_uploadFinished()));
  58. connect(m_reply,SIGNAL(uploadProgress(qint64,qint64)),this,SLOT(do_uploadProgress(qint64,qint64)));
  59. #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  60. connect(m_reply,SIGNAL(error(QNetworkReply::NetworkError)),this,SIGNAL(signal_error(QNetworkReply::NetworkError)));
  61. #endif
  62. }
  63. /**
  64. * @brief 重载函数,上传文件,直接上传内存数据
  65. *
  66. * @param data 需要上传的数据
  67. * @param farPath 远程文件名
  68. */
  69. void QtFtp::putFile(const QByteArray& data,const QString& farPath)
  70. {
  71. m_isFinished = false;
  72. m_result = false;
  73. m_url.setPath(farPath);
  74. m_reply = m_manager.put(QNetworkRequest(m_url),data);
  75. /* 将信号进行一次转发 */
  76. connect(m_reply,SIGNAL(uploadProgress(qint64,qint64)),this,SIGNAL(signal_uploadProgress(qint64,qint64)));
  77. /* 连接信号和槽 */
  78. connect(m_reply,SIGNAL(finished()),this,SLOT(do_uploadFinished()));
  79. connect(m_reply,SIGNAL(uploadProgress(qint64,qint64)),this,SLOT(do_uploadProgress(qint64,qint64)));
  80. #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  81. connect(m_reply,SIGNAL(error(QNetworkReply::NetworkError)),this,SIGNAL(signal_error(QNetworkReply::NetworkError)));
  82. #endif
  83. }
  84. /**
  85. * @brief QtFtp::getFile
  86. * @param fileName 本机文件路径和文件名
  87. * @param path 目标文件路径
  88. */
  89. void QtFtp::getFile(const QString &fileName,const QString &srcPath)
  90. {
  91. m_isFinished = false;
  92. m_result = false;
  93. m_file.setFileName(fileName);
  94. if(!m_file.open(QFile::WriteOnly | QFile::Truncate))
  95. {
  96. LOG_WARN("open file failed , file path : " + fileName);
  97. return;
  98. }
  99. m_url.setPath(srcPath);
  100. m_reply = m_manager.get(QNetworkRequest(m_url));
  101. /* 转发信号 */
  102. connect(m_reply,SIGNAL(downloadProgress(qint64,qint64)),this,SIGNAL(signal_downloadProgress(qint64,qint64)));
  103. /* 连接信号和槽 */
  104. connect(m_reply,SIGNAL(finished()),this,SLOT(do_downloadFinished()));
  105. connect(m_reply,SIGNAL(downloadProgress(qint64,qint64)),this,SLOT(do_downloadProgress(qint64,qint64)));
  106. #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  107. connect(m_reply,SIGNAL(error(QNetworkReply::NetworkError)),this,SIGNAL(signal_error(QNetworkReply::NetworkError)));
  108. #endif
  109. }
  110. /* 下载文件,直接下载到内存,路径是相对路径,不包含前面的IP地址和端口 */
  111. void QtFtp::getFile(QByteArray& data,const QString& srcPath)
  112. {
  113. m_isFinished = false;
  114. m_result = false;
  115. m_downloadData = &data;
  116. m_url.setPath(srcPath);
  117. m_reply = m_manager.get(QNetworkRequest(m_url));
  118. /* 转发信号 */
  119. connect(m_reply,SIGNAL(downloadProgress(qint64,qint64)),this,SIGNAL(signal_downloadProgress(qint64,qint64)));
  120. /* 连接信号和槽 */
  121. connect(m_reply,SIGNAL(finished()),this,SLOT(do_downloadFinishedToData()));
  122. connect(m_reply,SIGNAL(downloadProgress(qint64,qint64)),this,SLOT(do_downloadProgress(qint64,qint64)));
  123. #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  124. connect(m_reply,SIGNAL(error(QNetworkReply::NetworkError)),this,SIGNAL(signal_error(QNetworkReply::NetworkError)));
  125. #endif
  126. }
  127. /**
  128. * @brief 等待完成,同时设置超时时间,超时后,断开reply的信号和槽的连接
  129. * 参数设置为0表示查询是否完成,设置为-1表示一直等待
  130. *
  131. * @param msecs 超时时间
  132. * @arg -1 一直等待,将会阻塞在这里
  133. * @arg 0 查询是否完成
  134. * @arg 正整数,等待完成的时间,设置完成之后再次设置,会覆盖之前的设置
  135. * @arg < -1,阻塞等待传入的值的绝对值时间
  136. * @return true 上传或下载完成
  137. * @return false 上传或下载进行中
  138. */
  139. bool QtFtp::waitFinished(int msecs)
  140. {
  141. if(msecs == 0)
  142. {
  143. return m_isFinished;
  144. }
  145. else if(msecs == -1)
  146. {
  147. QEventLoop loop;
  148. connect(this, &QtFtp::signal_uploadState, &loop, &QEventLoop::quit);
  149. connect(this, &QtFtp::signal_downloadState, &loop, &QEventLoop::quit);
  150. loop.exec();
  151. return true;
  152. }
  153. else if (msecs > 0)
  154. {
  155. m_timer.start(msecs);
  156. return m_isFinished;
  157. }
  158. else if(msecs < -1)
  159. {
  160. QEventLoop loop;
  161. connect(this, &QtFtp::signal_uploadState, &loop, &QEventLoop::quit);
  162. connect(this, &QtFtp::signal_downloadState, &loop, &QEventLoop::quit);
  163. m_timer.start(-msecs);
  164. connect(&m_timer, &QTimer::timeout, &loop, &QEventLoop::quit);
  165. loop.exec();
  166. return false;
  167. }
  168. return m_isFinished;
  169. }
  170. /**
  171. * @brief 获取结果
  172. *
  173. * @return int
  174. * @arg -1 上传或下载失败
  175. * @arg 0 上传或下载成功
  176. */
  177. bool QtFtp::getResult()
  178. {
  179. return m_result;
  180. }
  181. /* 上传完成 */
  182. void QtFtp::do_uploadFinished()
  183. {
  184. if(m_timer.isActive())
  185. {
  186. m_timer.stop();
  187. }
  188. if(m_reply->error() == QNetworkReply::NoError)
  189. {
  190. m_result = true;
  191. emit signal_uploadState(true);
  192. LOG_INFO("----- FTP upload success -----");
  193. } else
  194. {
  195. LOG_WARN("FTP upload failed : " + m_reply->errorString());
  196. emit signal_uploadState(false);
  197. }
  198. m_reply->deleteLater();
  199. m_reply = nullptr;
  200. m_isFinished = true;
  201. }
  202. /* 创建目录 */
  203. bool QtFtp::createDir(const QString& path)
  204. {
  205. m_isFinished = false;
  206. m_result = false;
  207. QNetworkRequest request;
  208. QUrl url;
  209. url.setHost(m_url.host());
  210. url.setPort(m_url.port());
  211. url.setUserName(m_url.userName());
  212. url.setPassword(m_url.password());
  213. url.setPath(path);
  214. LOG_DEBUG("create dir : " + url.path());
  215. request.setUrl(url);
  216. request.setRawHeader("Content-Type", "application/x-www-form-urlencoded");
  217. QByteArray data;
  218. data.append("command=MKD");
  219. m_reply = m_manager.post(request,data);
  220. /* 连接信号和槽 */
  221. connect(m_reply,SIGNAL(finished()),this,SLOT(do_finished()));
  222. #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  223. connect(m_reply,SIGNAL(error(QNetworkReply::NetworkError)),this,SIGNAL(signal_error(QNetworkReply::NetworkError)));
  224. #endif
  225. return true;
  226. }
  227. /**
  228. * @brief 完成,保存文件
  229. * 这里采用逐步读取文件内容,因为QByteArray最多存储大约2GB的数据
  230. *
  231. */
  232. void QtFtp::do_downloadFinished()
  233. {
  234. if(m_timer.isActive())
  235. {
  236. m_timer.stop();
  237. }
  238. if(!m_file.isOpen())
  239. {
  240. m_file.open(QFile::WriteOnly | QFile::Truncate);
  241. }
  242. if(m_reply->error() == QNetworkReply::NoError)
  243. {
  244. while(!m_reply->atEnd())
  245. {
  246. /* 每次读取100MB */
  247. QByteArray data = m_reply->read(104857600);
  248. quint64 writenBytes = m_file.write(data);
  249. LOG_DEBUG("write file : " + QString::number(writenBytes));
  250. if(writenBytes != data.size())
  251. {
  252. LOG_WARN("FTP download failed: write file failed");
  253. emit signal_downloadState(false);
  254. m_result = false;
  255. break;
  256. }
  257. }
  258. m_file.flush();
  259. emit signal_downloadState(true);
  260. m_result = true;
  261. LOG_DEBUG("----- FTP download success -----");
  262. }else
  263. {
  264. LOG_WARN("FTP download failed:" + m_reply->errorString());
  265. /* 下载失败就删除创建的文件 */
  266. m_file.remove();
  267. emit signal_downloadState(false);
  268. }
  269. m_file.close();
  270. m_reply->deleteLater();
  271. m_reply = nullptr;
  272. m_isFinished = true;
  273. }
  274. /* 下载完成,保存数据 */
  275. void QtFtp::do_downloadFinishedToData()
  276. {
  277. if(m_timer.isActive())
  278. {
  279. m_timer.stop();
  280. }
  281. if(m_reply->error() == QNetworkReply::NoError)
  282. {
  283. *m_downloadData = m_reply->readAll();
  284. emit signal_downloadState(true);
  285. m_result = true;
  286. LOG_DEBUG("----- FTP download success -----");
  287. }else
  288. {
  289. LOG_WARN("FTP download failed:" + m_reply->errorString());
  290. emit signal_downloadState(false);
  291. }
  292. m_downloadData = nullptr;
  293. m_reply->deleteLater();
  294. m_reply = nullptr;
  295. m_isFinished = true;
  296. }
  297. /* 上传进度 */
  298. void QtFtp::do_uploadProgress(qint64 bytesSent, qint64 bytesTotal)
  299. {
  300. LOG_DEBUG("sent / total : " + QString::number(bytesSent) + " / " + QString::number(bytesTotal));
  301. }
  302. /* 下载进度 */
  303. void QtFtp::do_downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
  304. {
  305. LOG_DEBUG("received / total : " + QString::number(bytesReceived) + " / " + QString::number(bytesTotal));
  306. }
  307. /* ftp其他任务完成槽函数 */
  308. void QtFtp::do_finished()
  309. {
  310. if(m_timer.isActive())
  311. {
  312. m_timer.stop();
  313. }
  314. m_isFinished = true;
  315. if(m_reply->error() == QNetworkReply::NoError)
  316. {
  317. LOG_INFO("----- FTP operation success -----");
  318. m_result = true;
  319. }else
  320. {
  321. LOG_WARN("FTP operation failed : " + m_reply->errorString());
  322. m_result = false;
  323. }
  324. m_reply->deleteLater();
  325. m_reply = nullptr;
  326. m_isFinished = true;
  327. }
  328. /* 定时器超时函数 */
  329. void QtFtp::do_timeout()
  330. {
  331. if(m_reply != nullptr)
  332. {
  333. m_reply->abort();
  334. m_reply->deleteLater();
  335. m_reply = nullptr;
  336. m_result = false;
  337. }
  338. emit signal_uploadState(false);
  339. emit signal_downloadState(false);
  340. }
  341. /* ftp发送错误槽函数 */
  342. #if defined (QT_VERSION) && QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
  343. void QtFtp::do_ftpReplyError(QNetworkReply::NetworkError error)
  344. {
  345. if(error == QNetworkReply::NoError)
  346. {
  347. LOG_INFO("FTP发送成功");
  348. }
  349. else
  350. {
  351. LOG_WARN("FTP发送失败:" + QString::number(error));
  352. }
  353. }
  354. #endif