Для протокола FTP вы можете использовать FtpWebRequest
класс из .NET Framework. Хотя он не имеет явной поддержки рекурсивных операций с файлами (включая загрузки). Вы должны осуществить рекурсию самостоятельно:
- Список удаленных каталогов
- Повторяйте записи, загружая файлы и возвращаясь в подкаталоги (перечисляя их снова и т. Д.)
Сложная задача - определить файлы из подкаталогов. Нет способа сделать это портативным способом с FtpWebRequest
. К сожалению, FtpWebRequest
не поддерживает команду MLSD
, которая является единственным переносимым способом получения списка каталогов с атрибутами файлов в протоколе FTP. См. Также Проверка, является ли объект на FTP-сервере файлом или каталогом .
Ваши варианты:
- Выполните операцию с именем файла, которая наверняка не удастся для файла и удастся для каталогов (или наоборот). То есть Вы можете попробовать скачать «имя». Если это удается, это файл, если это не удается, это каталог. Но это может стать проблемой производительности, когда у вас есть большое количество записей.
- Возможно, вам повезет, и в вашем конкретном случае вы можете отличить файл из каталога по имени файла (то есть все ваши файлы имеют расширение, а подкаталоги - нет)
- Вы используете длинный список каталогов (
LIST
command = ListDirectoryDetails
метод) и пытаетесь проанализировать специфичный для сервера список. Многие FTP-серверы используют листинг в стиле * nix, где вы идентифицируете каталог по d
в самом начале записи. Но многие серверы используют другой формат. В следующем примере используется этот подход (в формате * nix)
void DownloadFtpDirectory(string url, NetworkCredential credentials, string localPath)
{
FtpWebRequest listRequest = (FtpWebRequest)WebRequest.Create(url);
listRequest.UsePassive = true;
listRequest.Method = WebRequestMethods.Ftp.ListDirectoryDetails;
listRequest.Credentials = credentials;
List<string> lines = new List<string>();
using (WebResponse listResponse = 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')
{
Directory.CreateDirectory(localFilePath);
DownloadFtpDirectory(fileUrl + "/", credentials, localFilePath);
}
else
{
FtpWebRequest downloadRequest = (FtpWebRequest)WebRequest.Create(fileUrl);
downloadRequest.UsePassive = true;
downloadRequest.UseBinary = true;
downloadRequest.Method = WebRequestMethods.Ftp.DownloadFile;
downloadRequest.Credentials = credentials;
using (Stream ftpStream = downloadRequest.GetResponse().GetResponseStream())
using (Stream fileStream = File.Create(localFilePath))
{
ftpStream.CopyTo(fileStream);
}
}
}
}
url
должно быть как:
Или используйте стороннюю библиотеку, которая поддерживает рекурсивные загрузки.
Например, с сборкой WinSCP .NET вы можете загрузить весь каталог за один вызов Session.GetFiles
:
// Setup session options
SessionOptions sessionOptions = new SessionOptions
{
Protocol = Protocol.Ftp,
HostName = "example.com",
UserName = "user",
Password = "mypassword",
};
using (Session session = new Session())
{
// Connect
session.Open(sessionOptions);
// Download files
session.GetFiles("/home/user/*", @"d:\download\").Check();
}
Внутренне WinSCP использует команду MLSD
, если она поддерживается сервером. Если нет, он использует команду LIST
и поддерживает десятки различных форматов списков.
(я автор WinSCP)