通过UDP来在两台服务器之间传输文件。
- 第一步要创建一个client:
client, err := sftp.NewClient("127.0.0.1:22345")
- 发送文件到服务端则:
sentBytes, err := client.SendFile(path)
- path为要发送的文件路径;可选参数offset(为断点续传预留)表示从文件的偏移量开始,默认从0开始;可选参数isRemove表示发送文件完成之后是否需要删除文件,默认不删除。
- error:文件不存在;服务端不存在;服务端拒绝;服务端超时;服务端故障;
- sentBytes: 表示的是服务端实际已经存在的偏移量,可能跟客户端本次发送的字节数不一致。
- 从服务端接收文件则:TODO
-
客户端发起一个读或写请求,请求内容包括:
- 读或写。
- 文件信息。文件名,文件大小,文件MD5等。
-
服务端收到请求,响应一个ack包,如果是读请求,包内容包括:
- 读请求。
- 文件是否存在,是否有权限读取。如果不存在或没有权限,同志客户端,并直接退出此次请求。
- 如果请求中包含文件的大小以及MD5值,则根据文件大小计算服务器上的文件的MD5是否一致,如果一致,则响应文件ackSame,并返回文件的全部大小以及MD5值。否则,返回ackNSame。
- 文件信息。文件名,如果服务器存在此文件,则返回此文件的信息,文件大小,文件MD5值等。
-
如果是写请求,包内容包括:
- 写请求。
- 文件是否存在,是否有权限写。如果不存在或没有权限,同志客户端,并直接退出此次请求。
- 如果文件存在,则根据文件大小计算服务器上的文件的MD5是否一致,如果一致,则响应文件ackSame,并返回文件的全部大小以及MD5值。否则,返回ackNSame。
-
客户端收到响应之后,如果是读请求,响应ack code应该是ackSame,ackNSame,ackNPermit,ackNExist中的一种:
- ackSame。回一个确认包,包内包含文件数据的起始位置(断点续传)。
- ackNSame。回一个确认包,包内包含是否继续传输。
- ackNPermit。放弃传输,不回确认包。
- ackNExist。放弃传输,不回确认包。
-
如果是写请求,响应ack code应该是ackSame,ackNSame,ackNPermit,ackNExist中的一种:
- ackSame。返回写成功。回一个确认包,包内包含文件数据的起始位置(断点续传)。
- ackNSame。根据服务端返回的文件大小和文件MD5值计算文件。回一个确认包,包内包含文件发送起始位置。
- ackNPermit。放弃传输,不回确认包。
- ackNExist。进入下一环节,开始发送数据。
-
服务器如果在等待响应。则根据以上具体响应内容决定是否继续传输以及传输开始位置。
启动两个goroutine,一个用来发数据,一个用来收数据的确认。
模仿TCP,实现一个弱化版的窗口机制。
blockNum*blockSize是窗口大小。
- 重传次数:retries
- 超时时间:timeout
不同于tcp的时延确认,每个数据包都要确认一次。
Inspired by tftp
tftp implemented by pin is very good. I don't intend to modify it, but just to add some controls on UDP to assure file tranfer.
Document will given later...
Code is in progress, so it will continue evolving little by little and at this point I'm not really looking for contributions.