我正在使用 Renci.SshNet 库通过 SFTP 递归获取文件和目录列表。我可以连接 SFTP 站点,但我不确定如何在 C# 中递归获取目录和文件列表。我还没有找到任何有用的例子。
有人试过这个吗?如果是这样,您能否发布一些有关如何递归获取这些文件和文件夹的示例代码。
谢谢,
普拉夫
这个库有一些使这个递归列表变得棘手的怪癖,因为 和 之间的交互ChangeDirectory
并不ListDirectory
像您预期的那样工作。
以下不列出/home 目录中的文件,而是列出 /(根)目录中的文件:
sftp.ChangeDirectory("home");
sftp.ListDirectory("").Select (s => s.FullName);
以下不起作用并返回 SftpPathNotFoundException:
sftp.ChangeDirectory("home");
sftp.ListDirectory("home").Select (s => s.FullName);
以下是列出/home目录下文件的正确方法
sftp.ChangeDirectory("/");
sftp.ListDirectory("home").Select (s => s.FullName);
如果你问我,这太疯狂了。使用方法设置默认目录对ChangeDirectory
方法没有影响,ListDirectory
除非您在此方法的参数中指定文件夹。似乎应该为此编写一个错误。
因此,当您编写递归函数时,您必须设置一次默认目录,然后在ListDirectory
遍历文件夹时更改调用中的目录。该清单返回一个可枚举的 SftpFiles。然后可以单独检查这些IsDirectory == true
。请注意,该列表还返回.
and..
条目(它们是目录)。如果你想避免无限循环,你会想跳过这些。:-)
编辑 2/23/2018
我正在查看我的一些旧答案,并想为上面的答案道歉并提供以下工作代码。请注意,此示例不需要ChangeDirectory
,因为它使用Fullname
for ListDirectory
:
void Main()
{
using (var client = new Renci.SshNet.SftpClient("sftp.host.com", "user", "password"))
{
var files = new List<String>();
client.Connect();
ListDirectory(client, ".", ref files);
client.Disconnect();
files.Dump();
}
}
void ListDirectory(SftpClient client, String dirName, ref List<String> files)
{
foreach (var entry in client.ListDirectory(dirName))
{
if (entry.IsDirectory)
{
ListDirectory(client, entry.FullName, ref files);
}
else
{
files.Add(entry.FullName);
}
}
}
尝试这个:
var filePaths = client.ListDirectory(client.WorkingDirectory);
我已经使用递归实现了这一点。像这样创建了一个类 TransportResponse
public class TransportResponse
{
public string directoryName { get; set; }
public string fileName { get; set; }
public DateTime fileTimeStamp { get; set; }
public MemoryStream fileStream { get; set; }
public List<TransportResponse> lstTransportResponse { get; set; }
}
我创建了一个 TransportResponse 类的列表。如果 directoryName 不为空,它将包含同一个类的列表,该类将在该目录中的文件作为 MemoryStream (这可以根据您的用例进行更改)
List<TransportResponse> lstResponse = new List<TransportResponse>();
using (var client = new SftpClient(connectionInfo))
{
try
{
Console.WriteLine("Connecting to " + connectionInfo.Host + " ...");
client.Connect();
Console.WriteLine("Connected to " + connectionInfo.Host + " ...");
}
catch (Exception ex)
{
Console.WriteLine("Could not connect to "+ connectionInfo.Host +" server. Exception Details: " + ex.Message);
}
if (client.IsConnected)
{
var files = client.ListDirectory(transport.SourceFolder);
lstResponse = downloadFilesInDirectory(files, client);
client.Disconnect();
}
else
{
Console.WriteLine("Could not download files from "+ transport.TransportIdentifier +" because client was not connected.");
}
}
private static List<TransportResponse> downloadFilesInDirectory(IEnumerable<SftpFile> files, SftpClient client)
{
List<TransportResponse> lstResponse = new List<TransportResponse>();
foreach (var file in files)
{
if (!file.IsDirectory)
{
if (file.Name != "." && file.Name != "..")
{
if (!TransportDAL.checkFileExists(file.Name, file.LastWriteTime))
{
using (MemoryStream fs = new MemoryStream())
{
try
{
Console.WriteLine("Reading " + file.Name + "...");
client.DownloadFile(file.FullName, fs);
fs.Seek(0, SeekOrigin.Begin);
lstResponse.Add(new TransportResponse { fileName = file.Name, fileTimeStamp = file.LastWriteTime, fileStream = new MemoryStream(fs.GetBuffer()) });
}
catch(Exception ex)
{
Console.WriteLine("Error reading File. Exception Details: " + ex.Message);
}
}
}
else
{
Console.WriteLine("File was downloaded previously");
}
}
}
else
{
if (file.Name != "." && file.Name != "..")
{
lstResponse.Add(new TransportResponse { directoryName = file.Name,lstTransportResponse = downloadFilesInDirectory(client.ListDirectory(file.Name), client) });
}
}
}
return lstResponse;
}
希望这可以帮助。谢谢
这是一个完整的课程。它是 .NET Core 2.1 Http 触发函数应用 (v2)
我想摆脱任何以“。”开头的目录,因为我的 sftp 服务器有 .cache 文件夹和带有键的 .ssh 文件夹。也不想处理像“。”这样的文件夹名称。或者 '..'
我最终要做的是将 SftpFile 投影到我使用的类型并将其返回给调用者(在这种情况下它将是一个逻辑应用程序)。然后我将该对象传递到一个存储过程中,并使用 OPENJSON 来构建我的监控表。这基本上是创建我的 SFTP 处理队列的第一步,该队列会将文件从我的 SFTP 文件夹中移到我的数据湖中(现在是 blob,直到我想出更好的东西)。
我使用 .WorkingDirectory 的原因是因为我创建了一个主目录为“/home”的用户。这让我可以遍历我的所有用户文件夹。我的应用程序不需要有一个特定的文件夹作为起点,可以说是用户“root”。
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.Extensions.Logging;
using Renci.SshNet;
using Renci.SshNet.Sftp;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
namespace SFTPFileMonitor
{
public class GetListOfFiles
{
[FunctionName("GetListOfFiles")]
public async Task<IActionResult> RunAsync([HttpTrigger(AuthorizationLevel.Anonymous, "get")] HttpRequest req, ILogger log)
{
log.LogInformation("C# HTTP trigger function processed a request.");
List<SftpFile> zFiles;
int fileCount;
decimal totalSizeGB;
long totalSizeBytes;
using (SftpClient sftpClient = new SftpClient("hostname", "username", "password"))
{
sftpClient.Connect();
zFiles = await GetFiles(sftpClient, sftpClient.WorkingDirectory, new List<SftpFile>());
fileCount = zFiles.Count;
totalSizeBytes = zFiles.Sum(l => l.Length);
totalSizeGB = BytesToGB(totalSizeBytes);
}
return new OkObjectResult(new { fileCount, totalSizeBytes, totalSizeGB, zFiles });
}
private async Task<List<SftpFile>> GetFiles(SftpClient sftpClient, string directory, List<SftpFile> files)
{
foreach (SftpFile sftpFile in sftpClient.ListDirectory(directory))
{
if (sftpFile.Name.StartsWith('.')) { continue; }
if (sftpFile.IsDirectory)
{
await GetFiles(sftpClient, sftpFile.FullName, files);
}
else
{
files.Add(sftpFile);
}
}
return files;
}
private decimal BytesToGB(long bytes)
{
return Convert.ToDecimal(bytes) / 1024 / 1024 / 1024;
}
}
}
@卡洛斯·博斯
这个库有一些使这个递归列表变得棘手的怪癖,因为 ChangeDirectory 和 ListDirectory 之间的交互不像你所期望的那样工作。
正确的
当 ChangeDirectory() 参数为 "." 时效果很好。
但如果你这样做
SftpClient sftp ...;
sftp.ChangeDirectory("some_folder");
//get file list
List<SftpFile> fileList = sftp.ListDirectory("some_folder").ToList();
然后有一个断言,因为 ListDirectory() 调用需要“some_folder/some_folder”
我使用的解决方法是在远程上传/重命名为“some_folder”之前保存并恢复当前目录,并且您需要在操作之前列出该文件夹(例如查看文件已经存在)
string working_directory = sftp.WorkingDirectory;
sftp.ChangeDirectory("some_folder");
sftp.RenameFile("name", "new_name");
sftp.ChangeDirectory(working_directory);
检查文件是否存在,这个调用就足够了
sftp.Exists(path)
或者如果您想添加一些其他条件,例如区分大小写或不区分大小写
public FileExistence checkFileExists(string folder, string fileName)
{
//get file list
List<SftpFile> fileList = sftp.ListDirectory(folder).ToList();
if (fileList == null)
{
return FileExistence.UNCONFIRMED;
}
foreach (SftpFile f in fileList)
{
Console.WriteLine(f.ToString());
//a not case sensitive comparison is made
if (f.IsRegularFile && f.Name.ToLower() == fileName.ToLower())
{
return FileExistence.EXISTS;
}
}
//if not found in traversal , it does not exist
return FileExistence.DOES_NOT_EXIST;
}
FileExistence 在哪里
public enum FileExistence
{
EXISTS,
DOES_NOT_EXIST,
UNCONFIRMED
};