FTP一般流程
FTP对应PASV和PORT两种访问方式,分别为被动和主动,是针对FTP服务器端进行区分的,正常传输过程中21号端口用于指令传输,数据传输端口使用其他端口。
PASV:由客户端发起数据传输请求,服务器端返回并携带数据端口,并且服务器端开始监听此端口等待数据,为被动模式;
PORT:客户端监听端口并向服务器端发起请求,服务器端主动连接此端口进行数据传输,为主动模式。
其中TYPE分两种模式,I对应二进制模式、A对应ASCII模式;
PASV为客户端发送请求,之后227为服务器端返回操作码表示成功,并且后面带有服务器端监听的端口:143x256(左移8位)+48
之后通过STOR命令进行数据下载,下载完成后返回226表示数据传输完成。
2. Python代码实现:
中文路径问题:由于FTP支持ASCII编码,Python ftplib中编码方式使用latin-1,而window默认编码方式为gbk,所以使用Python处理时需先将中文路径编码为gbk之后译码为latin-1字符;
上传下载使用storline和retrline,对应二进制使用storbinary和retrbinary。对于stor类函数后面的参数fp表示接收一个文件对象,支持read方法,一般为打开需要上传的源文件,而retr类函数后面的参数表示对于返回数据的处理方法。
从一个FTP服务器到另一个FTP服务器的数据传输:
利用本地电脑作为数据缓存,但并不将数据保存到硬盘,只在内存中存储进行数据传输;其中一端作为下载一端为数据上传。
首先登陆两个FTP服务器,transfercmd函数用于发送命令并返回已建立好连接的本地Socket,此时分别在两个本地Socket进行数据的收发即可。
在测试中发现,在发送完一个文件之后只有及时的关闭socket,21端口才会返回226,数据完成指示,这样才可以循环下一个文件,在完成之后要退出FTP。
#coding=utf-8 import ftplib,os.path,os import socket f1=ftplib.FTP('172.16.2.76') f2=ftplib.FTP('172.16.25.153') class myFTP: path='file/download/bbb/' # ftplib中编码使用latin-1 title='版本'.encode(encoding='gbk').decode(encoding='latin-1') path1=path+title localDir='E:\\ver\\fp\\' path2='abc/edf/' def __init__(self): try: f1.login('username','password') except ftplib.error_perm: print('f1 cannot loggin!') return try: f2.login() except ftplib.error_perm: print('f2 cannot loggin!') return def ftpD(self): filelist=[] fileLIST=[] filels=f1.retrlines('LIST %s'%(self.path1),callback=filelist.append) f1.cwd(self.path1) for file in filelist: fileAttr=file.split(' ') fileName=fileAttr[-1] fileType=fileAttr[0][0] if fileType=='-': fileLIST.append(fileName) for file in fileLIST: path=self.localDir+file f1.retrbinary('RETR %s'%(file),open(path,'wb').write) print('%s download.....'%(file)) f1.quit() def ftpU(self,fun=1): os.chdir(self.localDir) fileList=os.listdir() # upload file if fun==1: for file in fileList: path=self.path2 f2.storbinary('STOR %s'%(path+file),open(file,'rb')) print('%s uploading......'%(file)) #delete file if fun==0: try: for file in fileList: path=self.path2 f2.delete(path+file) print('%s delete......'%(file)) except ftplib.error_perm: print('no file to delete!!') return f2.quit() def test(self): f1.cwd(self.path1) f2.cwd(self.path2) fileList=f1.nlst() print(fileList) for file in fileList: print('transfer %s......'%(file)) f1.voidcmd('TYPE I') f2.voidcmd('TYPE I') sock1=f1.transfercmd('RETR %s'%(file)) sock2=f2.transfercmd('STOR %s'%(file)) while 1: data=sock1.recv(1024) sock2.sendall(data) if len(data)==0: break # 数据发送完成后需关闭socket,服务器21端口才会有返回 sock1.close() sock2.close() res1=f1.getresp() #print('f1 %s'%(res1)) res2=f2.getresp() #print('f2 %s'%(res2)) f1.quit() f2.quit() if __name__=='__main__': ftptest=myFTP() ftptest.ftpU(0) #ftptest.test() #ftptest.ftpD()
Python,FTP,文件传输
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。