Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4943358
  • 博文数量: 1696
  • 博客积分: 10870
  • 博客等级: 上将
  • 技术积分: 18357
  • 用 户 组: 普通用户
  • 注册时间: 2007-03-30 15:16
文章分类
文章存档

2017年(1)

2016年(1)

2015年(1)

2013年(1)

2012年(43)

2011年(17)

2010年(828)

2009年(568)

2008年(185)

2007年(51)

分类: C/C++

2010-03-21 22:01:39

前面已经分析过了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的时候再仔细看一下相关的实现。

阅读(1569) | 评论(0) | 转发(1) |
给主人留下些什么吧!~~