分类:
2012-05-26 22:14:21
原文地址:FileZilla FTP服务器源代码分析13 作者:sinodragon21
前面已经分析过了FTP客户登录服务器的过程,现在来看一下常见的ls命令的处理过程。
用户在FTP客户端输入ls命令后,ftp.exe首先发出port请求给服务器,在CControlSocket的ParseCommand() 中被处理。
PORT命令的参数是形如:127.0.0.1.4.9,前4个表示客户端的IP地址,后两个根据规则4 * 256 + 9 = 1033,表示FTP客户端临时建立的用来与服务器建立数据连接的端口,例子所示为1033端口。
PORT命令的处理过程的代码中前面都是用来获取IP和临时端口的:
case COMMAND_PORT:
...
port += 256 *
_ttoi(args.Right(args.GetLength() - (i + 1))); // add ms byte to server
socket
ip = args.Left(i);
...
下面:
m_transferstatus.ip = ip;
m_transferstatus.port = port;
m_transferstatus.pasv
= 0;
Send(_T("200 Port command successful"));
break;
只是将FTP客
户端提供的临时端口记录到m_transferstatus中,然后发出200 Port command
successful,等待FTP客户端的下一个命令。由于用户输入的是ls命令,ftp.exe在PORT之后,发出NLST命令。
在case COMMAND_NLST的处理中,先是进行了一系列的参数、权限检查,一切OK后:
if
(!m_transferstatus.pasv) // 主动模式
{
...
}
else // 被动模式
{
...
}
由
于主动模式是缺省值,因此看一下里面的代码:
CTransferSocket *transfersocket = new
CTransferSocket(this);
m_transferstatus.socket = transfersocket;
transfersocket->Init(pResult,
TRANSFERMODE_NLST); // 只是一些参数的初始化
if (m_transferMode == mode_zlib)
// 传输方式是否使用压缩方式,缺省不使用,详细参见FTP规范
{
if
(!transfersocket->InitZLib(m_zlibLevel))
{
Send(_T("550
could not initialize zlib, please use MODE S instead"));
ResetTransferstatus();
break;
}
}
if (!CreateTransferSocket(transfersocket)) // 建立数据连接
break;
SendTransferinfoNotification(TRANSFERMODE_LIST, physicalDir,
logicalDir); // Use TRANSFERMODE_LIST instead of TRANSFERMODE_NLST.
Send(_T("150
Opening data channel for directory list."));
先看一下建立数据连接的代码:
BOOL
CControlSocket::CreateTransferSocket(CTransferSocket *pTransferSocket)
{
...
if
(pTransferSocket->Connect(m_transferstatus.ip,m_transferstatus.port)==0)
...
}
无
非是常规的socket方法建立连接,需要关注的是由服务主动发起连接,这正是主动模式的含义。我们先看完这一段,再看一下被动模式。
在CreateTransferSocket()完成后,调用:
SendTransferinfoNotification(TRANSFERMODE_LIST, physicalDir, logicalDir);
看一下里面:
void CControlSocket::SendTransferinfoNotification(const
char transfermode, const CStdString& physicalFile, const
CStdString& logicalFile, __int64 startOffset, __int64 totalSize)
{
t_connop
*op = new t_connop;
op->op = USERCONTROL_CONNOP_TRANSFERINIT;
op->userid
= m_userid;
t_connectiondata_transferinfo *conndata = new
t_connectiondata_transferinfo;
conndata->transferMode =
transfermode;
conndata->physicalFile = physicalFile;
conndata->logicalFile
= logicalFile;
conndata->startOffset = startOffset;
conndata->totalSize
= totalSize;
op->data = conndata;
m_pOwner->SendNotification(FSM_CONNECTIONDATA, (LPARAM)op);
}
可
见发送了一个消息给CServer,wParam参数是FSM_CONNECTIONDATA,表示这是跟connection相关的消息,lParam
带的参数是USERCONTROL_CONNOP_TRANSFERINIT,表示传输开始或结束,我回去看一下CServer中的
OnServerMessage()相关代码,在admin窗口的下面显示了将用传输的信息。
下面,
Send(_T("150 Opening data channel for directory list."));
发
送给FTP客户端数据连接创建的消息,真正的数据传输的任务是交给数据连接了,即CTransferSocket。
我们回到被动模式,如果是被动模式:
if (!m_transferstatus.pasv)
{
...
}
else
// 被动模式
{
...
m_transferstatus.socket->PasvTransfer();
}
看
一下PasvTransfer()的实现:
void CTransferSocket::PasvTransfer()
{
if(bAccepted)
if
(!m_bStarted)
InitTransfer(FALSE);
}
非常简单,由于是被动模式,即由客户端发起数据连接,因此CTransferSocket只需等待客户端的连接就可以了,下面分析 CTransferSocket的时候再仔细看一下相关的实现。