3

我目前正在制作一个小型 .NET 控制台应用程序,将我的一些文件自动备份到我的服务器上。我遇到的问题是我遇到了一些恶劣的天气,导致一些电力和网络中断。在此期间,我注意到我的文件中有很大一部分没有通过或损坏。我想知道是否有办法获取另一端文件夹的大小,并查看文件名、文件数和总目录大小是否匹配。我已经尝试过 WinSCP 和 NcFTP 作为传输文件的方法,但我还没有看到任何关于获得适当文件大小的信息。

这几乎是一个窗口到窗口的传输,所以如果有一个命令行参数可以通过 FTP 客户端返回一个大小,那就太好了。

4

3 回答 3

2

没有标准的 FTP 命令来检索目录大小。

您必须递归地迭代所有子目录和文件并对大小求和。

这对于 .NET framework/ 来说并不容易FtpWebRequest,因为它不支持该MLSD命令,这是在 FTP 协议中检索具有文件属性的目录列表的唯一可移植方式。

您所能做的就是使用LIST命令 ( ListDirectoryDetails) 并尝试解析特定于服务器的列表。许多 FTP 服务器使用 *nix 样式的列表。但是许多服务器使用不同的格式。以下示例使用 *nix 格式:

static long CalculateFtpDirectorySize(string url, NetworkCredential credentials)
{
    FtpWebRequest listRequest = (FtpWebRequest)WebRequest.Create(url);
    listRequest.Method = WebRequestMethods.Ftp.ListDirectoryDetails;
    listRequest.Credentials = credentials;

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

    using (var listResponse = (FtpWebResponse)listRequest.GetResponse())
    using (Stream listStream = listResponse.GetResponseStream())
    using (StreamReader listReader = new StreamReader(listStream))
    {
        while (!listReader.EndOfStream)
        {
            lines.Add(listReader.ReadLine());
        }
    }

    long result = 0;
    foreach (string line in lines)
    {
        string[] tokens =
            line.Split(new[] { ' ' }, 9, StringSplitOptions.RemoveEmptyEntries);
        string name = tokens[8];
        string permissions = tokens[0];

        string fileUrl = url + name;

        if (permissions[0] == 'd')
        {
            result += CalculateFtpDirectorySize(fileUrl + "/", credentials);
        }
        else
        {
            result += long.Parse(tokens[4]);
        }
    }

    return result;
}

像这样使用它:

var credentials = new NetworkCredential("username", "password");
long size = CalculateFtpDirectorySize("ftp://ftp.example.com/", credentials);

如果您的服务器使用 DOS/Windows 列表格式,请参阅 C# 类来解析 WebRequestMethods.Ftp.ListDirectoryDe​​tails FTP 响应


或者,您可以使用支持现代MLSD命令的第 3 方 FTP 客户端实现。

例如WinSCP .NET 程序集支持它。

它甚至有方便的Session.EnumerateRemoteFiles方法,这使得计算目录大小变得容易:

var opts = EnumerationOptions.AllDirectories;
var files = session.EnumerateRemoteFiles("/", null, opts);
long size = files.Select(fileInfo => fileInfo.Length).Sum();

完整的代码如下:

SessionOptions sessionOptions = new SessionOptions
{
    Protocol = Protocol.Ftp,
    HostName = "ftp.example.com",
    UserName = "username",
    Password = "password",
};

using (Session session = new Session())
{
    // Connect
    session.Open(sessionOptions);

    var opts = EnumerationOptions.AllDirectories;
    var files = session.EnumerateRemoteFiles("/", null, opts);
    long size = files.Select(fileInfo => fileInfo.Length).Sum();
}

(我是WinSCP的作者)

于 2017-09-04T19:29:59.877 回答
1

没有标准的方法来请求“此目录中文件的总大小”。您可以通过 单独询问每个文件大小SIZE file.txt,也可以询问ls -l整个目录并解析文件大小。

于 2011-10-25T16:58:57.757 回答
0

我认为最好的办法是获取完整的文件列表,然后一次发送一个。如果在传输过程中连接失败,则该文件上传将失败。如果文件上传成功,那么您可以从列表中删除该文件名。

正如您在标签中提到的 NET,也许您应该在这里查看如何在 C# 中执行它的示例。VB.Net 将是类似的,但它会给你一个想法。

您可以获得一个目录列表,如下所示

于 2011-10-26T07:17:05.673 回答