FileZilla FTP服務(wù)器源代碼分析(2)
FileZilla FTP服務(wù)器源代碼分析:FileZilla是一款免費(fèi)而且開(kāi)源的FTP工具。包括FileZilla Client,F(xiàn)ileZilla Server兩個(gè)版本。FileZilla Server只提供了windows系統(tǒng)下的版本,我們要將本地的網(wǎng)站網(wǎng)頁(yè)文件上傳到網(wǎng)站服務(wù)器,或從服務(wù)器下載網(wǎng)頁(yè)文件,只需FileZilla Client客戶端版本就可以了。FileZilla FTP服務(wù)器源代碼分。在進(jìn)一步分析代碼之前,先復(fù)習(xí)一下FTP協(xié)議,下圖是FTP的結(jié)構(gòu)圖。

客戶端和服務(wù)器是通過(guò)兩個(gè)連接來(lái)進(jìn)行通訊的:
一個(gè)是控制連接,也就是傳輸些控制命令,客戶端發(fā)出FTP命令,服務(wù)器給出應(yīng)答,例如:USER,PASS命令等等。這個(gè)連接中,F(xiàn)TP服務(wù)器的端 口就是熟知的21端口,連接是由客戶端發(fā)起的,例如:ftp 192.168.0.1。有一點(diǎn)注意,用戶是通過(guò)“用戶接口”來(lái)操作的,一般的用戶接口是指cuteFTP這些FTP客戶端,或者ftp.exe這種命令 行程序,用戶在用戶接口使用的是ftp命令,如ls, get, cd等,這些ftp命令并不是真正與FTP服務(wù)器交互的命令,這些ftp命令還需要由“用戶協(xié)議解釋器”翻譯成真正的ftp協(xié)議命令,如USER, PASS,才能與服務(wù)器進(jìn)行交互。
一個(gè)是數(shù)據(jù)連接,即真正的文件傳輸是在這個(gè)連接上進(jìn)行的。服務(wù)器端的數(shù)據(jù)連接端口是20,客戶端的數(shù)據(jù)連接端口是隨機(jī)生成的。數(shù)據(jù)連接只在傳輸文件 時(shí)存在,文件傳完后,這個(gè)連接就斷了,如果需要再次傳送文件,會(huì)再次建立一個(gè)數(shù)據(jù)連接(客戶端的端口是隨機(jī)的,不一定是上次的那個(gè))。數(shù)據(jù)連接的模式有兩 種,一種是主動(dòng)方式,一種是被動(dòng)方式,兩者的區(qū)別在于數(shù)據(jù)連接是由誰(shuí)發(fā)起。
我們來(lái)看一個(gè)典型的FTP交互過(guò)程,用的是windows的ftp.exe程序,先建立一個(gè)連接,然后ls看一下文件列表,用get命令下 載一個(gè)文件,***quit關(guān)閉。下面-d選項(xiàng)可以顯示交互的細(xì)節(jié),注意-->開(kāi)頭的行是ftp客戶端發(fā)給FTP服務(wù)器的請(qǐng)求,3個(gè)數(shù)字開(kāi)頭的行是服 務(wù)器的應(yīng)答,如220, 331等開(kāi)頭的行:
C:\>ftp -d localhost
Connected to dell.
220-FileZilla Server version 0.9.18 beta
220-written by Tim Kosse (Tim.Kosse@gmx.de)
220 Please visit http://sourceforge.net/projects/filezilla/
User (dell:(none)): robert
---> USER robert
331 Password required for robert
Password:
---> PASS test
230 Logged on
ftp> ls
---> PORT 127,0,0,1,4,173
200 Port command successful
---> NLST
150 Opening data channel for directory list.
Manual.txt
226 Transfer OK
ftp: 收到 175 字節(jié),用時(shí) 0.00Seconds 175000.00Kbytes/sec.
ftp> get Manual.txt
---> PORT 127,0,0,1,4,174
200 Port command successful
---> RETR Manual.txt
150 Opening data channel for file transfer.
226 Transfer OK
ftp: 收到 17319 字節(jié),用時(shí) 0.09Seconds 192.43Kbytes/sec.
ftp> quit
---> QUIT
221 Goodbye
C:\>
剛開(kāi)始,客戶端發(fā)出建立連接的請(qǐng)求:
- C:\>ftp -d localhost // 建立連接
- Connected to dell. // 連接已建立
然后服務(wù)器發(fā)送歡迎信息,并要求輸入用戶名:
- 220-FileZilla Server version 0.9.18 beta
- 220-written by Tim Kosse (Tim.Kosse@gmx.de)
- 220 Please visit http://sourceforge.net/projects/filezilla/
- User (dell:(none)):
客戶端輸入用戶名robert,然后回車(chē):
---> USER robert // ftp.exe生成FTP命令:USER,發(fā)送給服務(wù)器
服務(wù)器要求輸入密碼:
331 Password required for robert
Password:
客戶端輸入密碼,然后回車(chē):
---> PASS test // ftp.exe生成FTP命令:PASS,發(fā)送給服務(wù)器
服務(wù)器通過(guò)密碼驗(yàn)證:
230 Logged on
客戶端鍵入ls命令
ftp> ls
ftp.exe生成FTP命令:PORT,告訴服務(wù)器客戶端的隨機(jī)端口是什么
---> PORT 127,0,0,1,4,173 // 127,0,0,1是IP地址,4 * 256 + 173 = 1197是隨機(jī)端口號(hào)
200 Port command successful // 服務(wù)器響應(yīng)PORT命令
---> NLST // 客戶端發(fā)出NLST命令,要求列出文件列表
150 Opening data channel for directory list. // 服務(wù)器會(huì)在20端口與客戶端的1197端口建立數(shù)據(jù)連接,傳輸數(shù)據(jù),注意ls命令的結(jié)果是在“數(shù)據(jù)連接”中傳輸?shù)?/P>
Manual.txt // 只有一個(gè)文件
226 Transfer OK // FTP服務(wù)器響應(yīng),傳輸完畢
ftp: 收到 175 字節(jié),用時(shí) 0.00Seconds 175000.00Kbytes/sec. // FTP客戶端顯示的傳輸結(jié)果
下面客戶端要求下載Manual.txt文件
ftp> get Manual.txt
---> PORT 127,0,0,1,4,174 // 告訴服務(wù)器客戶端新的隨機(jī)端口4 * 256 + 174 = 1198
200 Port command successful // // 服務(wù)器響應(yīng)PORT命令
---> RETR Manual.txt // 告訴服務(wù)器下載Manual.txt文件
150 Opening data channel for file transfer. // 服務(wù)器會(huì)在20端口與客戶端的1198端口建立數(shù)據(jù)連接,傳輸數(shù)據(jù)
226 Transfer OK // FTP服務(wù)器響應(yīng),傳輸完畢
ftp: 收到 17319 字節(jié),用時(shí) 0.09Seconds 192.43Kbytes/sec. // FTP客戶端顯示的傳輸結(jié)果
***客戶端退出
ftp> quit
---> QUIT // 發(fā)出QUIT命令
221 Goodbye // 服務(wù)器***響應(yīng)
仔細(xì)閱讀上面的交互過(guò)程,可以發(fā)現(xiàn),用戶手工輸入的一個(gè)FTP命令,可能會(huì)被ftp.exe處理成與FTP服務(wù)器的多次交互。如ls, get命令。
要想詳細(xì)了解FTP命令的細(xì)節(jié),可以參見(jiàn)FTP的RFC,或者相關(guān)的資料,不過(guò)由于我們閱讀源代碼的主要目的不是研究FTP細(xì)節(jié),而在于掌握高并發(fā)的網(wǎng)絡(luò)編程的技術(shù),所以,我們只以上面這個(gè)簡(jiǎn)單的FTP交互來(lái)看一下,在代碼中這個(gè)過(guò)程是如何實(shí)現(xiàn)的。
【編輯推薦】