using System;
using System.Net;
using System.IO;
using System.Text;
using System.Net.Sockets;

namespace FTP


{

/**//// <summary>
/// 类描述:对FTP 操作类
/// 作者:何火生
/// </summary>
public class FTPFactory

{

Arrbuite#region Arrbuite
private string remoteHost, remotePath, remoteUser, remotePass, mes;
private int remotePort, bytes;
private Socket clientSocket;

private int retValue;
private Boolean debug;
private Boolean logined;
private string reply;
private static int BLOCK_SIZE = 512;
Byte[] buffer = new Byte[BLOCK_SIZE];
Encoding ASCII = Encoding.ASCII;
#endregion

protected FTPFactory()

{
remoteHost = "localhost";
remotePath = ".";
remoteUser = "anonymous";
remotePass = "jaimon@school2000.co.uk";
FTPSetup();
}


构造函数#region 构造函数
private void FTPSetup()

{
remotePort = 21;
debug = false;
logined = false;
}
#endregion


getFileList#region getFileList

/**//// <summary>
/// Return a string array containing the remote directory's file list.
/// </summary>
/// <param name="mask"></param>
/// <returns></returns>
public string[] getFileList(string mask)

{

if (!logined)

{
login();
}

Socket cSocket = createDataSocket();
sendCommand("NLST " + mask);
if (!(retValue == 150 || retValue == 125))

{
throw new IOException(reply.Substring(4));
}

mes = "";

while (true)

{

int bytes = cSocket.Receive(buffer, buffer.Length, 0);
mes += ASCII.GetString(buffer, 0, bytes);

if (bytes < buffer.Length)

{
break;
}
}


char[] seperator =
{ '\n' };
string[] mess = mes.Split(seperator);

cSocket.Close();

readReply();

if (retValue != 226)

{
throw new IOException(reply.Substring(4));
}
return mess;

}
#endregion


getFileSize#region getFileSize

/**//// <summary>
/// Return the size of a file.
/// </summary>
/// <param name="fileName"></param>
/// <returns></returns>
public long getFileSize(string fileName)

{

if (!logined)

{
login();
}

sendCommand("SIZE " + fileName);
long size = 0;

if (retValue == 213)

{
size = Int64.Parse(reply.Substring(4));
}
else

{
throw new IOException(reply.Substring(4));
}

return size;

}
#endregion


login#region login

/**//// <summary>
/// Login to the remote server.
/// </summary>
public void login()

{

clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint ep = new IPEndPoint(Dns.GetHostEntry(remoteHost).AddressList[0], remotePort);
try

{
clientSocket.Connect(ep);
}
catch

{
//throw new IOException("Couldn't connect to remote server");;
}

readReply();
if (retValue != 220)

{
Close();
//throw new IOException(reply.Substring(4));
}
if (debug)
Console.WriteLine("USER " + remoteUser);

sendCommand("USER " + remoteUser);
if (!(retValue == 331 || retValue == 230))

{
cleanup();
//throw new IOException(reply.Substring(4));
}

if (retValue != 230)

{
if (debug)
Console.WriteLine("PASS xxx");

sendCommand("PASS " + remotePass);
if (!(retValue == 230 || retValue == 202))

{
cleanup();
//throw new IOException(reply.Substring(4));
}
}
logined = true;
Console.WriteLine("Connected to " + remoteHost);
ChDir(remotePath);
}
#endregion


setBinaryMode#region setBinaryMode


/**//// <summary>
/// If the value of mode is true, set binary mode for downloads.
/// Else, set Ascii mode.
/// </summary>
/// <param name="mode"></param>
public void setBinaryMode(Boolean mode)

{

if (mode)

{
sendCommand("TYPE I");
}
else

{
sendCommand("TYPE A");
}
if (retValue != 200)

{
throw new IOException(reply.Substring(4));
}
}
#endregion


Download#region Download

/**//// <summary>
/// Download a file to the Assembly's local directory,
/// keeping the same file name.
/// </summary>
/// <param name="remFileName"></param>
/// <returns></returns>
public bool Download(string remFileName)

{
return Download(remFileName, "", false);
}


/**//// <summary>
/// Download a remote file to the Assembly's local directory,
/// keeping the same file name, and set the resume flag.
/// </summary>
/// <param name="remFileName"></param>
/// <param name="resume"></param>
/// <returns></returns>
public bool Download(string remFileName, Boolean resume)

{
return Download(remFileName, "", resume);
}


/**//// <summary>
/// Download a remote file to a local file name which can include
/// a path. The local file name will be created or overwritten,
/// but the path must exist.
/// </summary>
/// <param name="remFileName"></param>
/// <param name="locFileName"></param>
/// <returns></returns>
public bool Download(string remFileName, string locFileName)

{
return Download(remFileName, locFileName, false);
}



/**//// <summary>
/// Download a remote file to a local file name which can include
/// a path, and set the resume flag. The local file name will be
/// created or overwritten, but the path must exist.
/// </summary>
/// <param name="remFileName"></param>
/// <param name="locFileName"></param>
/// <param name="resume"></param>
/// <returns></returns>
public bool Download(string remFileName, string locFileName, Boolean resume)

{

bool Finish = false;
bool FileExists = false;
if (!logined)

{
login();
}
setBinaryMode(true);
try

{
FileInfo fileInfo = new FileInfo(locFileName);
if (fileInfo.Exists)

{
fileInfo.IsReadOnly = false;
fileInfo.Delete();
}
}
catch

{
return false;
}
foreach (string str in getFileList("."))

{
if (str.IndexOf(remFileName.ToUpper(), StringComparison.CurrentCultureIgnoreCase) != -1)

{
FileExists = true;
break;
}
}
if (!FileExists)

{
//Console.WriteLine("文件不存在!");
return false;
}
Console.WriteLine("Downloading file " + remFileName + " from " + remoteHost + "/" + remotePath);

if (locFileName.Equals(""))

{
locFileName = remFileName;
}

if (!File.Exists(locFileName))

{
Stream st = File.Create(locFileName);
st.Close();
}
FileStream output = null;
try

{
output = new FileStream(locFileName, FileMode.Open);
}
catch (Exception ex)

{
Console.WriteLine(ex.Message);
return false;
}
Socket cSocket = createDataSocket();
long offset = 0;
if (resume)

{
offset = output.Length;
if (offset > 0)

{
sendCommand("REST " + offset);
if (retValue != 350)

{
//throw new IOException(reply.Substring(4));
//Some servers may not support resuming.
Finish = false;
offset = 0;
}
}

if (offset > 0)

{
if (debug)

{
Console.WriteLine("seeking to " + offset);
}
long npos = output.Seek(offset, SeekOrigin.Begin);
Console.WriteLine("new pos=" + npos);
}
}

sendCommand("RETR " + remFileName);
if (!(retValue == 150 || retValue == 125))

{
throw new IOException(reply.Substring(4));
}

while (true)

{
bytes = cSocket.Receive(buffer, buffer.Length, 0);
output.Write(buffer, 0, bytes);
if (bytes <= 0)

{
Finish = true;
break;
}
}

output.Close();
if (cSocket.Connected)

{
cSocket.Close();
}
readReply();
if (!(retValue == 226 || retValue == 250))

{
throw new IOException(reply.Substring(4));
}
return Finish;
}
#endregion


Upload#region Upload

/**//// <summary>
/// Upload a file
/// </summary>
/// <param name="fileName"></param>
/// <returns>Upload a file is success!</returns>
public bool Upload(string fileName)

{
return Upload(fileName, false);
}



/**//// <summary>
/// Upload a file and set the resume flag.
/// </summary>
/// <param name="fileName"></param>
/// <param name="resume"></param>
/// <returns></returns>
public bool Upload(string fileName, Boolean resume)

{
if (!logined)

{
login();
}
Socket cSocket = createDataSocket();
long offset = 0;
if (resume)

{
try

{
setBinaryMode(true);
offset = getFileSize(fileName);

}
catch (Exception)

{
offset = 0;
}
}

if (offset > 0)

{
sendCommand("REST " + offset);
if (retValue != 350)

{
//throw new IOException(reply.Substring(4));
//Remote server may not support resuming.
offset = 0;
}
}

sendCommand("STOR " + Path.GetFileName(fileName));

if (!(retValue == 125 || retValue == 150))

{
throw new IOException(reply.Substring(4));
}

// open input stream to read source file
FileStream input = new FileStream(fileName, FileMode.Open);
if (offset != 0)

{

if (debug)

{
Console.WriteLine("seeking to " + offset);
}
input.Seek(offset, SeekOrigin.Begin);
}

Console.WriteLine("Uploading file " + fileName + " to " + remotePath);

while ((bytes = input.Read(buffer, 0, buffer.Length)) > 0)

{
cSocket.Send(buffer, bytes, 0);
}
input.Close();
Console.WriteLine("");
if (cSocket.Connected)

{
cSocket.Close();
}
readReply();
if (!(retValue == 226 || retValue == 250))

{
throw new IOException(reply.Substring(4));
}
return true;
}
#endregion


DeleteRemoteFile#region DeleteRemoteFile

/**//// <summary>
/// Delete a file from the remote FTP server.
/// </summary>
/// <param name="fileName"></param>
public void DeleteRemoteFile(string fileName)

{

if (!logined)

{
login();
}

sendCommand("DELE " + fileName);
if (retValue != 250)

{
throw new IOException(reply.Substring(4));
}

}
#endregion


RenameRemoteFile#region RenameRemoteFile

/**//// <summary>
/// Rename a file on the remote FTP server.
/// </summary>
/// <param name="oldFileName"></param>
/// <param name="newFileName"></param>
public void RenameRemoteFile(string oldFileName, string newFileName)

{

if (!logined)

{
login();
}

sendCommand("RNFR " + oldFileName);

if (retValue != 350)

{
throw new IOException(reply.Substring(4));
}

// known problem
// rnto will not take care of existing file.
// i.e. It will overwrite if newFileName exist
sendCommand("RNTO " + newFileName);
if (retValue != 250)

{
throw new IOException(reply.Substring(4));
}

}
#endregion


MkDir#region MkDir

/**//// <summary>
/// Create a directory on the remote FTP server.
/// </summary>
/// <param name="dirName"></param>
public void MkDir(string dirName)

{

if (!logined)

{
login();
}

sendCommand("MKD " + dirName);

if (retValue != 250)

{
throw new IOException(reply.Substring(4));
}

}
#endregion


RmDir#region RmDir

/**//// <summary>
/// Delete a directory on the remote FTP server
/// </summary>
/// <param name="dirName"></param>
public void RmDir(string dirName)

{

if (!logined)

{
login();
}

sendCommand("RMD " + dirName);

if (retValue != 250)

{
throw new IOException(reply.Substring(4));
}

}
#endregion


ChDir#region ChDir

/**//// <summary>
/// Change the current working directory on the remote FTP server.
/// </summary>
/// <param name="dirName"></param>
public void ChDir(string dirName)

{

if (dirName.Equals("."))

{
return;
}

if (!logined)

{
login();
}

sendCommand("CWD " + dirName);

if (retValue != 250)

{
throw new IOException(reply.Substring(4));
}

this.remotePath = dirName;

Console.WriteLine("Current directory is " + remotePath);

}
#endregion


Close()#region Close()

/**//// <summary>
/// Close the FTP connection.
/// </summary>
public void Close()

{

if (clientSocket != null)

{
sendCommand("QUIT");
}

cleanup();
Console.WriteLine("Closing
");
}
#endregion


OTHER#region OTHER

/**//// <summary>
/// Set debug mode.
/// </summary>
/// <param name="debug"></param>
public void setDebug(Boolean debug)

{
this.debug = debug;
}

private void readReply()

{
mes = "";
reply = readLine();
retValue = Int32.Parse(reply.Substring(0, 3));
}

private void cleanup()

{
if (clientSocket != null)

{
clientSocket.Close();
clientSocket = null;
}
logined = false;
}
#endregion


readLine#region readLine
private string readLine()

{

while (true)

{
bytes = clientSocket.Receive(buffer, buffer.Length, 0);
mes += ASCII.GetString(buffer, 0, bytes);
if (bytes < buffer.Length)

{
break;
}
}


char[] seperator =
{ '\n' };
string[] mess = mes.Split(seperator);

if (mes.Length > 2)

{
mes = mess[mess.Length - 2];
}
else

{
mes = mess[0];
}

if (!mes.Substring(3, 1).Equals(" "))

{
return readLine();
}

if (debug)

{
for (int k = 0; k < mess.Length - 1; k++)

{
Console.WriteLine(mess[k]);
}
}
return mes;
}
#endregion


sendCommand#region sendCommand
private void sendCommand(String command)

{

Byte[] cmdBytes = Encoding.ASCII.GetBytes((command + "\r\n").ToCharArray());
clientSocket.Send(cmdBytes, cmdBytes.Length, 0);
readReply();
}
#endregion


createDataSocket#region createDataSocket
private Socket createDataSocket()

{
sendCommand("PASV");
if (retValue != 227)

{
throw new IOException(reply.Substring(4));
}

int index1 = reply.IndexOf('(');
int index2 = reply.IndexOf(')');
string ipData = reply.Substring(index1 + 1, index2 - index1 - 1);
int[] parts = new int[6];
int len = ipData.Length;
int partCount = 0;
string buf = "";
for (int i = 0; i < len && partCount <= 6; i++)

{

char ch = Char.Parse(ipData.Substring(i, 1));
if (Char.IsDigit(ch))
buf += ch;
else if (ch != ',')

{
throw new IOException("Malformed PASV reply: " + reply);
}

if (ch == ',' || i + 1 == len)

{

try

{
parts[partCount++] = Int32.Parse(buf);
buf = "";
}
catch (Exception)

{
throw new IOException("Malformed PASV reply: " + reply);
}
}
}
string ipAddress = parts[0] + "." + parts[1] + "." + parts[2] + "." + parts[3];
int port = (parts[4] << 8) + parts[5];
Socket s = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
IPEndPoint ep = new IPEndPoint(Dns.GetHostEntry(ipAddress).AddressList[0], port);
try

{
s.Connect(ep);
}
catch (Exception)

{
throw new IOException("Can't connect to remote server");
}
return s;
}
#endregion

}

}


posted on
2007-11-30 13:06
wskfire
阅读(
5529)
评论()
收藏
举报