实现FTP断点续传

应用需求:

网盘开发工作逐步进入各部分的整合阶段,当用户在客户端修改或新增加一个文件时,该文件要同步上传到服务器端对应的用户目录下,因此针对数据传输(即:上传、下载)这一块现在既定了三种传输方式,即:Ftp传输、HTTP传输以及基于UDT的传输。且这三种数据传输方式是可配的,可以通过不同的接口调用。相比这三种方式,基于UDT的大量文件传输是比较值得研究与创新的地方,它在底层是基于UDP,在上层实现了可靠性的控制;同时它充分考虑到了基于在公网环境下基于Tcp进行传输时拥塞控制算法的缺点,实现了自己的拥塞控制算法,在实际测试中其性能也是明显高于基于Tcp的传输。关于UDT实现文件传输只进行了技术调研,还没有真正实现,这一部分内容将在后续文章中提及。这三天的时间只实现了基于FTP的支持断点续传的文件上传、下载。

实现原理:

离我们最近的断点续传的应用例子是:迅雷。当使用迅雷下载一个大文件时,它实现了下面的功能:1> 电脑突然断电或程序突然退出后,当我们重新启动迅雷时它还会从程序退出时已经下载的文件点继续向后下载,而不是文件又从头开始下载。2> 可以设置采用多个线程同时下载,每个线程只下载文件中的某一部分,例如:使用三个线程下载一个9000个字节的文件,则第一个线程下载第1—3000个字节,第二个线程下载第3001—6000个字节,第三个线程下载第6001—9000个字节。这三个线程是同时下载一个文件,只是下载不同的部分,它会把下载的文件片段暂存在某个位置,当三个线程全部下载完成时再拼成一个完整的文件。这里不用多说,其优点显而易见。

其实,断点续传实现的原理很简单,就是无论是上传还是下载时都可以实时记录下已经上传了或下载了多少字节,如果中间因为某种原因传输断开,下载启动时只需要再重新从已经下载的位置继续下载或上传就可以了。

利用Qftp实现断点续传:

QT中有一个实现Ftp的类:Qftp,它提供了基本的ftp的使用方式,连接ftp服务器:connectToHost;登录:login;上传:put;下载:get,使用这些方法可以实现与ftp服务器交互实现文件上传、下载。但是使用它原生提供的put与get方法,无法实现断点续传。因此,为了实现断点续传我们需要重新实现文件传输,并在其中添加断点续传的控制。其实Ftp文件传输的本质也是使用Tcp来实现底层的文件传输。大体思路就是:利用Qftp的connectToHost登录ftp服务器,使用login登录ftp服务器,使用rawCommand发送ftp原生态的命令,使用QTcpSocket实现文件数据的传输。

首先,使用QTcpSocket实现文件数据的传输,需要设置ftp服务器为“PASV”被动接收方式,即ftp服务器被动地接收来自客户端的连接请求。

Ftp服务器所有可以发送的原生命令有:http://www.nsftools.com/tips/RawFTP.htm。

实现断点上传的命令发送流程:

1、rawCommand("TYPE I");设置传输数据的类型:二进制数据或ASCII

2、rawCommand("PASV");设置服务器为被动接收方式。发送PASV命令后,服务器会返回自己开启的数据传输的端口,等待客户端连接进行数据传输。返回的数据格式为:“227 Entering Passive Mode (192, 168, 2, 18, 118, 32)”,然后从返回的信息里面或去相关的信息,ftp服务器的IP地址:192.168.2.18;ftp服务器开启的用于数据传输的端口:118*256 + 32 = 30240;获得该信息后就需要建立TcpSocket的通信链路,连接ftp服务器。

3、rawCommand("APPE  remote-file-path");设置服务器端remote-file-path为追加的方式。如果此时改文件不存在,则服务器端会创建一个。

4、完成上述流程后,就可以打开本地文件进行读取,并通过tcpsocket链路发送出去(write)。

实现断点下载的命令发送流程:

1、rawCommand("TYPE I");设置传输数据的类型:二进制数据或ASCII

2、rawCommand("PASV");设置服务器为被动接收方式。发送PASV命令后,服务器会返回自己开启的数据传输的端口,等待客户端连接进行数据传输。返回的数据格式为:“227 Entering Passive Mode (192, 168, 2, 18, 118, 32)”,然后从返回的信息里面或去相关的信息,ftp服务器的IP地址:192.168.2.18;ftp服务器开启的用于数据传输的端口:118*256 + 32 = 30240;获得该信息后就需要建立TcpSocket的通信链路,连接ftp服务器。

3、rawCommand("REST  size");该命令设置ftp服务器从本地文件的哪个地方开始进行数据传输。

4、rawCommand(“RETR  remote-file-path”);开始从远程主机传输文件。

文件上传时在设置APPE返回之后,就可以打开本地文件进行上传;文件下载时,收到PASV的返回信息建立tcpsocket的连接后,需要建立readyRead()的信号槽,在该槽函数中实现数据的读取。

关键代码:

1. 处理rawCommand()发送原生命令返回后的槽函数:

[cpp] view plaincopy

  1. void LHTFileTransfer::ProcRawCommandReply(int nReplyCode, QString strDetail)
  2. {
  3. //! TYPE
  4. if (200 == nReplyCode)
  5. {
  6. m_ftpHandle->rawCommand("PASV");
  7. if (currentItem.task_type.compare("Upload") == 0)
  8. {
  9. op = QString("Put");
  10. }
  11. else if (currentItem.task_type.compare("DownLoad") == 0)
  12. {
  13. op = QString("Get");
  14. }
  15. }
  16. //! PASV
  17. else if(227 == nReplyCode)
  18. {
  19. const QString backResult = strDetail;
  20. if (NULL != m_sendDataSocket)
  21. {
  22. m_sendDataSocket->close();
  23. delete m_sendDataSocket;
  24. }
  25. m_sendDataSocket = new QTcpSocket();
  26. connect(m_sendDataSocket, SIGNAL(readyRead()), this, SLOT(ProcReadyRead()), Qt::UniqueConnection);
  27. connect(m_sendDataSocket, SIGNAL(readChannelFinished()), this, SLOT(ProcReadChannelFinished()), Qt::UniqueConnection);
  28. connect(m_sendDataSocket, SIGNAL(bytesWritten(qint64)), this, SLOT(ProcBytesWritten(qint64)), Qt::UniqueConnection);
  29. QStringList lstr = backResult.split("(").last().split(")").first().split(",");
  30. int nAddress = lstr.at(0).toInt()<<24 |
  31. lstr.at(1).toInt()<<16 |
  32. lstr.at(2).toInt()<<8 |
  33. lstr.at(3).toInt();
  34. QHostAddress hostAddress(nAddress);
  35. int nPort = lstr.at(lstr.length() - 2).toInt() * 256 + lstr.last().toInt();
  36. m_sendDataSocket->connectToHost(hostAddress, nPort);
  37. //! APPE , 需要接远程文件的绝对路径
  38. QString appeShell;
  39. if (op.compare("Put") == 0)
  40. {
  41. appeShell = QString("APPE %1").arg(currentItem.file_remote_path);
  42. }
  43. else if (op.compare("Get") == 0)
  44. {
  45. //! 这里的REST后面的大小应该为本地保存的问价的大小
  46. appeShell = QString("REST 0");
  47. }
  48. m_ftpHandle->rawCommand(appeShell);
  49. }
  50. //! 发送数据
  51. else if (150 == nReplyCode)
  52. {
  53. if (op.compare("Put") == 0)
  54. {
  55. m_fileHandle = new QFile(currentItemFilePath);
  56. if (!m_fileHandle->open(QIODevice::ReadOnly))
  57. {
  58. qDebug() << "file open error ...";
  59. return ;
  60. }
  61. const qint64 fileSize = m_fileHandle->size();
  62. m_fileHandle->seek(currentItem.uploaded_size);
  63. while(!m_fileHandle->atEnd())
  64. {
  65. const qint64 nBlockSize = 16 * 1024 ;
  66. char buf[16 * 1024];
  67. qint64 nowPos = m_fileHandle->pos();
  68. qint64 readLen = m_fileHandle->read(buf, nBlockSize);
  69. if (readLen !=0 && readLen != -1)
  70. {
  71. m_sendDataSocket->write(buf, readLen);
  72. m_sendDataSocket->flush();
  73. emit DataTransferProgress(nowPos, fileSize);
  74. }
  75. }
  76. m_sendDataSocket->flush();
  77. m_sendDataSocket->close();
  78. m_sendDataSocket = NULL ;
  79. emit DataTransferProgress(m_fileHandle->pos(), m_fileHandle->size());
  80. m_procTask.remove(currentItemFilePath);
  81. m_fileHandle->close();
  82. //emit StartNextTask();
  83. }
  84. else if(op.compare("Get") == 0)
  85. {
  86. m_fileHandle = new QFile(currentItem.file_remote_path);
  87. if (!m_fileHandle->open(QIODevice::WriteOnly))
  88. {
  89. qDebug() << "file open error ...";
  90. return ;
  91. }
  92. }
  93. }
  94. else if(350 == nReplyCode)
  95. {
  96. QString shell = QString("RETR %1").arg(currentItemFilePath);
  97. m_ftpHandle->rawCommand(shell);
  98. }
  99. }

2. 断点下载时实现buffer读取的函数:

[cpp] view plaincopy

  1. void LHTFileTransfer::ProcReadyRead()
  2. {
  3. qDebug() << "[DownLoad] ProcReadyReady ....";
  4. QByteArray buffer = m_sendDataSocket->readAll();
  5. m_fileHandle->write(buffer);
  6. m_fileHandle->flush();
  7. emit DataTransferProgress(m_fileHandle->size(), 0);
  8. }

面临的问题以及后续需要优化的地方:

1. 字符编码问题,即当需要上传的文件名是中文名称时,需要对其进行转码。

2. 现在实现的是单线程,尚未添加多线程断点下载以及队列的实现。

时间: 2025-01-05 19:15:10

实现FTP断点续传的相关文章

【大话QT之十】实现FTP断点续传

应用需求: 网盘开发工作逐步进入各部分的整合阶段,当用户在客户端修改或新增加一个文件时,该文件要同步上传到服务器端对应的用户目录下,因此针对数据传输(即:上传.下载)这一块现在既定了三种传输方式,即:Ftp传输.HTTP传输以及基于UDT的传输.且这三种数据传输方式是可配的,可以通过不同的接口调用.相比这三种方式,基于UDT的大量文件传输是比较值得研究与创新的地方,它在底层是基于UDP,在上层实现了可靠性的控制:同时它充分考虑到了基于在公网环境下基于Tcp进行传输时拥塞控制算法的缺点,实现了自己

Python学习笔记——进阶篇【第八周】———FTP断点续传作业&amp;批量主机管理工具

主机管理之paramiko模块学习 http://www.cnblogs.com/wupeiqi/articles/5095821.html 作业1:用socketserver继续完善FTP作业 作业2:开发一个批量主机管理工具 需求: 可以对机器进行分组 可以对指定的一组或多组机器执行批量命令,分发文件(发送\接收) 纪录操作日志 作业参考链接http://www.cnblogs.com/alex3714/articles/5227251.html

python学习之路(三)使用socketserver进行ftp断点续传

最近学习python到socketserver,本着想试一下水的深浅,采用Python3.6. 目录结构如下: receive_file和file为下载或上传文件存放目录,ftp_client为ftp客户端,ftp_server为server端. server端源码: #!/usr/bin/env python # -*- coding:utf-8 -*- import socketserver import os error_code = {'400':'FILE IS NOT EXISTS'

ftp断点续传

有时候ftp的文件太大了 容易断掉 使用shell下载 1 #!/bin/bash 2 cd /data2/GATK2/refSeqDB/1000genomePhase3 3 ftp -v -n 193.62.192.8 <<EOF 4 user ftp '' 5 binary 6 cd vol1 7 cd ftp 8 cd release 9 cd 20130502 10 prompt off 11 mget * 12 close 13 bye 14 EOF

edtftpj让Java上传FTP文件支持断点续传

在用Java实现FTP上传文件功能时,特别是上传大文件的时候,可以需要这样的功能:程序在上传的过程中意外终止了,文件传了一大半,想从断掉了地方继续传:或者想做类似迅雷下载类似的功能,文件太大,今天传一半,睡一觉去先,明天继续传. Java上传FTP文件,用的比较多的工具是apache的commons-net.如果想用commons-net实现FTP上传的断点续传还是有点麻烦. 除了commons-net之外,还有很多非常优秀的FTP工具,这里使用edtftpj这个工具来实现断点续传. 下载:ht

TCP/IP协议原理与应用笔记02:断点续传

1.断点续传简介:       FTP(文件传输协议的简称)(File Transfer Protocol. FTP)客户端软件断点续传指的是在下载或上传时,将下载或上传任务(一个文件或一个压缩包)人为的划分为几个部分,每一个部分采用一个线程进行上传或下载,如果碰到网络故障,可以从已经上传或下载的部分开始继续上传下载未完成的部分,而没有必要从头开始上传下载.用户可以节省时间,提高速度. 2.用途 有时用户上传下载文件需要历时数小时,万一线路中断,不具备断点续传的FTP服务器或下载软件就只能从头重

No.4 文件断点续传(Client)

import socket, json, struct, hashlib, os, shutil codes = { "文件已存在":1001, "文件不存在":1002 } ''' 后开始 1.接收字典的长度: 2.利用字典的长度来接收字典(FileName,File_md5值,FileSize) 3.利用File_md5值判断需下载的文件是否存在.(若存在,则需要续传) 4.若文件已存在发送字典(Code,FileSize) 若文件不存在发送字典(Code,Fi

Python常用模块-SYS、OS、Time、hashlib

MarkdownPad Document 时间模块 import time 三种时间表达形式: 1.时间戳 time.time()  相对于1970年到现在的秒数 2.格式化的时间字符串 time.strftime("%Y-%m-%d %X") -可以是:可以是/ '2017-04-26 00:32:18' 3.时间元组(结构化时间) time.localtime() time.struct_time(tm_year=2017, tm_mon=4, tm_mday=26, tm_hou

curl错误码

curl错误码列表: 1    curl 不支持该协议    2    curl 初始化失败    3    URL 格式错误    5    解析代理服务器失败    6    解析主机失败    7    建立与主机的连接失败    8    无法解析 FTP 服务器返回的消息    9    FTP 服务器拒接访问.可能是拒绝登录或拒绝访问特定目录,但很多情况下是访问了一个不存在的位置导致的    11    无法解析 FTP 服务器的 PASS 回复消息    13    无法解析 FT