C#通过FTP下载所有文件和子目录

我仍在学习C#。为了帮助自己,我正在尝试创建一个程序,该程序将自动将所有本地项目与FTP服务器上的文件夹同步。这样一来,无论我是在学校还是在家,我总是可以使用相同的项目。

我知道有些像Dropbox这样的程序已经为我完成了这项工作,但是我认为创建类似的东西会让自己在整个过程中学到很多。

我实现目标的第一步是仅从FTP服务器下载所有文件,子目录和子文件。我设法使用下面的代码从目录下载了所有文件。但是,我的代码仅列出了文件夹名称和主目录中的文件。子文件夹和子文件永远不会返回,也不会下载。除此之外,服务器返回550错误,因为我试图像下载文件一样下载文件夹。我已经花了4个小时以上的时间,但是我只是找不到解决这些问题并使之工作的方法。因此,我希望你们能帮助我:)

public string[] GetFileList()

{

string[] downloadFiles;

StringBuilder result = new StringBuilder();

WebResponse response = null;

StreamReader reader = null;

try

{

FtpWebRequest request = (FtpWebRequest)WebRequest.Create(url);

request.UseBinary = true;

request.Method = WebRequestMethods.Ftp.ListDirectory;

request.Credentials = new NetworkCredential(ftpUserName, ftpPassWord);

request.KeepAlive = false;

request.UsePassive = false;

response = request.GetResponse();

reader = new StreamReader(response.GetResponseStream());

string line = reader.ReadLine();

while (line != null)

{

result.Append(line);

result.Append("\n");

line = reader.ReadLine();

}

result.Remove(result.ToString().LastIndexOf('\n'), 1);

return result.ToString().Split('\n');

}

catch (Exception ex)

{

if (reader != null)

{

reader.Close();

}

if (response != null)

{

response.Close();

}

downloadFiles = null;

return downloadFiles;

}

}

private void Download(string file)

{

try

{

string uri = url + "/" + file;

Uri serverUri = new Uri(uri);

if (serverUri.Scheme != Uri.UriSchemeFtp)

{

return;

}

FtpWebRequest request = (FtpWebRequest)WebRequest.Create(url + "/" + file);

request.UseBinary = true;

request.Method = WebRequestMethods.Ftp.DownloadFile;

request.Credentials = new NetworkCredential(ftpUserName, ftpPassWord);

request.KeepAlive = false;

request.UsePassive = false;

FtpWebResponse response = (FtpWebResponse)request.GetResponse();

Stream responseStream = response.GetResponseStream();

FileStream writeStream = new FileStream(localDestnDir + "\\" + file, FileMode.Create);

int Length = 2048;

Byte[] buffer = new Byte[Length];

int bytesRead = responseStream.Read(buffer, 0, Length);

while (bytesRead > 0)

{

writeStream.Write(buffer, 0, bytesRead);

bytesRead = responseStream.Read(buffer, 0, Length);

}

writeStream.Close();

response.Close();

}

catch (WebException wEx)

{

MessageBox.Show(wEx.Message, "Download Error");

}

catch (Exception ex)

{

MessageBox.Show(ex.Message, "Download Error");

}

}

回答:

FtpWebRequest没有递归文件操作(包括下载)任何明确的支持。您必须自己实现递归:

  • 列出远程目录
  • 迭代条目,下载文件并递归到子目录(再次列出它们,等等)

棘手的部分是从子目录中识别文件。使用。不能以可移植的方式执行此操作FtpWebRequest。该FtpWebRequest遗憾的是不支持的MLSD命令,这是检索目录与FTP协议的文件属性上市的唯一可移植的方法。另请参阅检查FTP服务器上的对象是文件还是目录。

您的选择是:

  • 对一定要对文件失败但对目录成功的文件名执行操作(反之亦然)。即,您可以尝试下载“名称”。如果成功,则为文件;如果失败,则为目录。
  • 您可能很幸运,在您的特定情况下,您可以通过文件名告诉目录中的文件(即,所有文件都有扩展名,而子目录则没有)
  • 您使用长目录列表(LISTcommand = ListDirectoryDetailsmethod)并尝试解析服务器特定的列表。许多FTP服务器使用 nix样式的列表,您可以d在条目的开始处通过来标识目录。但是许多服务器使用不同的格式。下面的示例使用这种方法(假设 nix格式)

    void DownloadFtpDirectory(string url, NetworkCredential credentials, string localPath)

    {

    FtpWebRequest listRequest = (FtpWebRequest)WebRequest.Create(url);

    listRequest.Method = WebRequestMethods.Ftp.ListDirectoryDetails;

    listRequest.Credentials = credentials;

    List<string> lines = new List<string>();

    using (FtpWebResponse listResponse = (FtpWebResponse)listRequest.GetResponse())

    using (Stream listStream = listResponse.GetResponseStream())

    using (StreamReader listReader = new StreamReader(listStream))

    {

    while (!listReader.EndOfStream)

    {

    lines.Add(listReader.ReadLine());

    }

    }

    foreach (string line in lines)

    {

    string[] tokens =

    line.Split(new[] { ' ' }, 9, StringSplitOptions.RemoveEmptyEntries);

    string name = tokens[8];

    string permissions = tokens[0];

    string localFilePath = Path.Combine(localPath, name);

    string fileUrl = url + name;

    if (permissions[0] == 'd')

    {

    if (!Directory.Exists(localFilePath))

    {

    Directory.CreateDirectory(localFilePath);

    }

    DownloadFtpDirectory(fileUrl + "/", credentials, localFilePath);

    }

    else

    {

    FtpWebRequest downloadRequest = (FtpWebRequest)WebRequest.Create(fileUrl);

    downloadRequest.Method = WebRequestMethods.Ftp.DownloadFile;

    downloadRequest.Credentials = credentials;

    using (FtpWebResponse downloadResponse =

    (FtpWebResponse)downloadRequest.GetResponse())

    using (Stream sourceStream = downloadResponse.GetResponseStream())

    using (Stream targetStream = File.Create(localFilePath))

    {

    byte[] buffer = new byte[10240];

    int read;

    while ((read = sourceStream.Read(buffer, 0, buffer.Length)) > 0)

    {

    targetStream.Write(buffer, 0, read);

    }

    }

    }

    }

    }

使用如下功能:

NetworkCredential credentials = new NetworkCredential("user", "mypassword");

string url = "ftp://ftp.example.com/directory/to/download/";

DownloadFtpDirectory(url, credentials, @"C:\target\directory");


如果要避免解析特定于服务器的目录列表格式的麻烦,请使用支持该MLSD命令和/或解析各种LIST列表格式的第三方库。和递归下载。

例如,使用WinSCP

.NET程序集,您可以通过一次调用来下载整个目录Session.GetFiles

// Setup session options

SessionOptions sessionOptions = new SessionOptions

{

Protocol = Protocol.Ftp,

HostName = "ftp.example.com",

UserName = "user",

Password = "mypassword",

};

using (Session session = new Session())

{

// Connect

session.Open(sessionOptions);

// Download files

session.GetFiles("/directory/to/download/*", @"C:\target\directory\*").Check();

}

MLSD如果服务器支持,WinSCP在内部使用该命令。如果没有,它将使用该LIST命令并支持多种不同的列表格式。

默认情况下,该Session.GetFiles方法是递归的。

(我是WinSCP的作者)

以上是 C#通过FTP下载所有文件和子目录 的全部内容, 来源链接: utcz.com/qa/412116.html

回到顶部