FTP 命令
FTP 的主要操作都是基于各种命令基础之上的。常用的命令有:
设置传输模式,它包括ASCⅡ(文本) 和BINARY 二进制模式
目录操作,改变或显示远程计算机的当前目录(cd、dir/ls 命令)
连接操作,open命令用于建立同远程计算机的连接;close命令用于关闭连接
发送操作,put命令用于传送文件到远程计算机;mput 命令用于传送多个文件到远程计算机
获取操作,get命令用于接收一个文件;mget命令用于接收多个文件。
?
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635
import java.net.Socketimport org.apache.log4j.Logger/** * 角色——服务器A * @author Leon * */public class ServerA{ public static void main(String[] args){final String F_DIR = "c:/test"//根路径final int PORT = 22//监听端口号Logger.getRootLogger() Logger logger = Logger.getLogger("com") try{ ServerSocket s = new ServerSocket(PORT) logger.info("Connecting to server A...") logger.info("Connected Successful! Local Port:"+s.getLocalPort()+". Default Directory:'"+F_DIR+"'.") while( true ){//接受客户端请求Socket client = s.accept() //创建服务线程new ClientThread(client, F_DIR).start() }} catch(Exception e) { logger.error(e.getMessage()) for(StackTraceElement ste : e.getStackTrace()){logger.error(ste.toString()) }} }}import java.io.BufferedReaderimport java.io.Fileimport java.io.FileNotFoundExceptionimport java.io.IOExceptionimport java.io.InputStreamimport java.io.InputStreamReaderimport java.io.OutputStreamimport java.io.PrintWriterimport java.io.RandomAccessFileimport java.net.ConnectExceptionimport java.net.InetAddressimport java.net.ServerSocketimport java.net.Socketimport java.net.UnknownHostExceptionimport java.nio.charset.Charsetimport java.util.Randomimport org.apache.log4j.Logger/** * 客户端子线程类 * @author Leon * */public class ClientThread extends Thread { private Socket socketClient//客户端socket private Logger logger//日志对象 private String dir//绝对路径 private String pdir = "/"//相对路径 private final static Random generator = new Random()//随机数 public ClientThread(Socket client, String F_DIR){this.socketClient = client this.dir = F_DIR } @Override public void run() {Logger.getRootLogger() logger = Logger.getLogger("com") InputStream is = null OutputStream os = null try { is = socketClient.getInputStream() os = socketClient.getOutputStream() } catch (IOException e) { logger.error(e.getMessage()) for(StackTraceElement ste : e.getStackTrace()){logger.error(ste.toString()) }}BufferedReader br = new BufferedReader(new InputStreamReader(is,Charset.forName("UTF-8"))) PrintWriter pw = new PrintWriter(os) String clientIp = socketClient.getInetAddress().toString().substring(1)//记录客户端IPString username = "not logged in"//用户名String password = ""//口令String command = ""//命令boolean loginStuts = false//登录状态final String LOGIN_WARNING = "530 Please log in with USER and PASS first." String str = ""//命令内容字符串int port_high = 0 int port_low = 0 String retr_ip = ""//接收文件的IP地址Socket tempsocket = null //打印欢迎信息pw.println("220-FTP Server A version 1.0 written by Leon Guo") pw.flush() logger.info("("+username+") ("+clientIp+")>Connected, sending welcome message...") logger.info("("+username+") ("+clientIp+")>220-FTP Server A version 1.0 written by Leon Guo") boolean b = true while ( b ){ try {//获取用户输入的命令command = br.readLine() if(null == command) break } catch (IOException e) {pw.println("331 Failed to get command") pw.flush() logger.info("("+username+") ("+clientIp+")>331 Failed to get command") logger.error(e.getMessage()) for(StackTraceElement ste : e.getStackTrace()){ logger.error(ste.toString()) }b = false } /* * 访问控制命令 */ // USER命令 if(command.toUpperCase().startsWith("USER")){logger.info("(not logged in) ("+clientIp+")>"+command) username = command.substring(4).trim()if("".equals(username)){ pw.println("501 Syntax error") pw.flush() logger.info("(not logged in) ("+clientIp+")>501 Syntax error") username = "not logged in" }else{ pw.println("331 Password required for " + username) pw.flush() logger.info("(not logged in) ("+clientIp+")>331 Password required for " + username) }loginStuts = false } //end USER // PASS命令 else if(command.toUpperCase().startsWith("PASS")){logger.info("(not logged in) ("+clientIp+")>"+command) password = command.substring(4).trim()if(username.equals("root") &&password.equals("root")){ pw.println("230 Logged on") pw.flush() logger.info("("+username+") ("+clientIp+")>230 Logged on")// logger.info("客户端 "+clientIp+" 通过 "+username+"用户登录") loginStuts = true }else{ pw.println("530 Login or password incorrect!") pw.flush() logger.info("(not logged in) ("+clientIp+")>530 Login or password incorrect!") username = "not logged in" } } //end PASS // PWD命令 else if(command.toUpperCase().startsWith("PWD")){logger.info("("+username+") ("+clientIp+")>"+command) if(loginStuts){// logger.info("用户"+clientIp+":"+username+"执行PWD命令") pw.println("257 /""+pdir+"/" is current directory") pw.flush() logger.info("("+username+") ("+clientIp+")>257 /""+pdir+"/" is current directory") }else{ pw.println(LOGIN_WARNING) pw.flush() logger.info("("+username+") ("+clientIp+")>"+LOGIN_WARNING) } } //end PWD // CWD命令 else if(command.toUpperCase().startsWith("CWD")){logger.info("("+username+") ("+clientIp+")>"+command) if(loginStuts){ str = command.substring(3).trim() if("".equals(str)){pw.println("250 Broken client detected, missing argument to CWD. /""+pdir+"/" is current directory.") pw.flush() logger.info("("+username+") ("+clientIp+")>250 Broken client detected, missing argument to CWD. /""+pdir+"/" is current directory.") } else{//判断目录是否存在String tmpDir = dir + "/" + str File file = new File(tmpDir) if(file.exists()){//目录存在 dir = dir + "/" + str if("/".equals(pdir)){pdir = pdir + str } else{pdir = pdir + "/" + str }// logger.info("用户"+clientIp+":"+username+"执行CWD命令") pw.println("250 CWD successful. /""+pdir+"/" is current directory") pw.flush() logger.info("("+username+") ("+clientIp+")>250 CWD successful. /""+pdir+"/" is current directory") }else{//目录不存在 pw.println("550 CWD failed. /""+pdir+"/": directory not found.") pw.flush() logger.info("("+username+") ("+clientIp+")>550 CWD failed. /""+pdir+"/": directory not found.")
你用的FTPClient引入不对吧,我们项目上都是用的
import org.apache.commons.net.ftp.FTPClient
import org.apache.commons.net.ftp.FTPFile
import org.apache.commons.net.ftp.FTPReply
下面是我们项目上用到的FTP的实现代码(FTP需要先连接,再登录,之后就是校验登录是否成功),具体代码如下:
/*** 获取FTPClient对象
*
* @param ftpHost FTP主机服务器
* @param ftpPassword FTP 登录密码
* @param ftpUserName FTP登录用户名
* @param ftpPort FTP端口 默认为21
* @return FTPClient
* @throws Exception
*/
public static FTPClient getFTPClient(String ftpHost, String ftpUserName,
String ftpPassword, int ftpPort) throws Exception {
try {
FTPClient ftpClient = new FTPClient()
ftpClient.connect(ftpHost, ftpPort)// 连接FTP服务器
ftpClient.login(ftpUserName, ftpPassword)// 登陆FTP服务器
if (!FTPReply.isPositiveCompletion(ftpClient.getReplyCode())) {
logger.error("未连接到FTP,用户名或密码错误!")
ftpClient.disconnect()
return null
} else {
logger.info("FTP连接成功!")
return ftpClient
}
} catch (SocketException socketException) {
logger.error("FTP的IP地址可能错误,请正确配置!")
throw socketException
} catch (IOException ioException) {
logger.error("FTP的端口错误,请正确配置!")
throw ioException
}
}
一.FTP的PORT(主动模式)和PASV(被动模式)1.
PORT(主动模式)
PORT中文称为主动模式,工作的原理:
FTP客户端连接到FTP服务器的21端口,发送用户名和密码登录,登录成功后要list列表或者读取数据时,客户端随机开放一个端口(1024以上),发送
PORT命令到FTP服务器,告诉服务器客户端采用主动模式并开放端口;FTP服务器收到PORT主动模式命令和端口号后,通过服务器的20端口和客户端开放的端口连接,发送数据.
2.
PASV(被动模式)
PASV是Passive的缩写,中文成为被动模式,工作原理:FTP客户端连接到FTP服务器的21端口,发送用户名和密码登录,登录成功后要list列表或者读取数据时,发送PASV命令到FTP服务器,
服务器在本地随机开放一个端口(1024以上),然后把开放的端口告诉客户端,
客户端再连接到服务器开放的端口进行数据传输。
二.两种模式的比较
从上面的运行原来看到,主动模式和被动模式的不同简单概述为:
主动模式传送数据时是“服务器”连接到“客户端”的端口;被动模式传送数据是“客户端”连接到“服务器”的端口。
主动模式需要客户端必须开放端口给服务器,很多客户端都是在防火墙内,开放端口给FTP服务器访问比较困难。
被动模式只需要服务器端开放端口给客户端连接就行了。
三.不同工作模式的网络设置
实际项目中碰到的问题是,FTP的客户端和服务器分别在不同网络,两个网络之间有至少4层的防火墙,服务器端只开放了21端口,
客户端机器没开放任何端口。FTP客户端连接采用的被动模式,结果客户端能登录成功,但是无法LIST列表和读取数据。很明显,是因为服务器端没开放被动模式下的随机端口导致。
由于被动模式下,服务器端开放的端口随机,但是防火墙要不能全部开放,解决的方案是,在ftp服务器配置被动模式下开放随机端口在
50000-60000之间(范围在ftp服务器软件设置,可以设置任意1024上的端口段),然后在防火墙设置规则,开放服务器端50000-60000之间的端口端。
主动模式下,客户端的FTP软件设置主动模式开放的端口段,在客户端的防火墙开放对应的端口段。
四.如何设置
工作模式
实时上FTP服务器一般都支持主动和被动模式,连接采用何种模式是有FTP客户端软件决定。
欢迎分享,转载请注明来源:夏雨云
评论列表(0条)