Загрузка файла с FTP с помощью Progress - TotalBytesToReceive всегда равен -1? - PullRequest
7 голосов
/ 04 января 2011

Я пытаюсь загрузить файл с FTP-сервера с индикатором выполнения.

Файл загружается, и вызывается событие ProgressChanged, за исключением того, что в аргументах события TotalBytesToReceive всегда равно -1. TotalBytes увеличивается, но я не могу рассчитать процент без общей суммы.

Я полагаю, что могу найти размер файла с помощью других команд ftp, но мне интересно, почему это не работает?

Мой код:

FTPClient request = new FTPClient();
request.Credentials = credentials;
request.DownloadProgressChanged += new DownloadProgressChangedEventHandler(request_DownloadProgressChanged);
//request.DownloadDataCompleted += new DownloadDataCompletedEventHandler(request_DownloadDataCompleted);
request.DownloadDataAsync(new Uri(folder + file));
while (request.IsBusy) ;

....

static void request_DownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e)
{
    if (e.TotalBytesToReceive == -1)
    {
        l.reportProgress(-1, FormatBytes(e.BytesReceived) + " out of ?" );
    }
    else
    {
        l.reportProgress(e.ProgressPercentage, "Downloaded " + FormatBytes(e.BytesReceived) + " out of " + FormatBytes(e.TotalBytesToReceive) + " (" + e.ProgressPercentage + "%)");
    }
}

....

class FTPClient : WebClient
{
    protected override WebRequest GetWebRequest(System.Uri address)
    {
        FtpWebRequest req = (FtpWebRequest)base.GetWebRequest(address);
        req.UsePassive = false;
        return req;
    }
}

Спасибо.

Ответы [ 3 ]

4 голосов
/ 13 апреля 2013

Итак, у меня была такая же проблема.Я обошел его, сначала извлекая размер файла.

        // Get the object used to communicate with the server.
        FtpWebRequest request = (FtpWebRequest)WebRequest.Create("URL");
        request.Method = WebRequestMethods.Ftp.GetFileSize;
        request.Credentials = networkCredential;
        FtpWebResponse response = (FtpWebResponse)request.GetResponse();

        Stream responseStream = response.GetResponseStream();
        bytes_total = response.ContentLength; //this is an int member variable stored for later
        Console.WriteLine("Fetch Complete, ContentLength {0}", response.ContentLength);
        response.Close();

        webClient = new MyWebClient();
        webClient.Credentials = networkCredential; ;
        webClient.DownloadDataCompleted += new DownloadDataCompletedEventHandler(FTPDownloadCompleted);
        webClient.DownloadProgressChanged += new DownloadProgressChangedEventHandler(FTPDownloadProgressChanged);
        webClient.DownloadDataAsync(new Uri("URL"));

Затем выполняем математические вычисления в обратном вызове.

private void FTPDownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e)
{
     progressBar.Value = (int)(((float)e.BytesReceived / (float)bytes_total) * 100.0);
}
2 голосов
/ 13 августа 2017

При использовании протокола FTP WebClient в общем случае не знает общий размер загрузки.Таким образом, вы обычно получаете -1 с FTP.

Обратите внимание, что поведение на самом деле противоречит документации .NET, в которой говорится для FtpWebResponse.ContentLength (откуда взято значение TotalBytesToReceive):

Для запросов, использующих метод DownloadFile, свойство больше нуля, если загруженный файл содержит данные, и равно нулю, если оно пустое.

Новы легко узнаете много вопросов по этому поводу, эффективно показывая, что поведение не всегда так задокументировано.FtpWebResponse.ContentLength имеет значащее значение только для метода GetFileSize.

FtpWebRequest / WebClient не делает явных попыток выяснить размер загружаемого файла.Все, что он делает, это пытается найти (xxx bytes). строку в 125 / 150 ответах на команду RETR.Нет RFC FTP требует, чтобы сервер включал такую ​​информацию.ProFTPD (см. data_pasv_open в src/data.c) и vsftpd (см. handle_retr в postlogin.c), по-видимому, включают эту информацию.Другие обычные FTP-серверы (IIS, FileZilla) этого не делают.


Если ваш сервер не предоставляет информацию о размере, вам необходимо самостоятельно запросить размер перед загрузкой.Полное решение с использованием FtpWebRequest и Task:

private void button1_Click(object sender, EventArgs e)
{
    // Run Download on background thread
    Task.Run(() => Download());
}

private void Download()
{
    try
    {
        const string url = "ftp://ftp.example.com/remote/path/file.zip";
        NetworkCredential credentials = new NetworkCredential("username", "password");

        // Query size of the file to be downloaded
        WebRequest sizeRequest = WebRequest.Create(url);
        sizeRequest.Credentials = credentials;
        sizeRequest.Method = WebRequestMethods.Ftp.GetFileSize;
        int size = (int)sizeRequest.GetResponse().ContentLength;

        progressBar1.Invoke(
            (MethodInvoker)(() => progressBar1.Maximum = size));

        // Download the file
        WebRequest request = WebRequest.Create(url);
        request.Credentials = credentials;
        request.Method = WebRequestMethods.Ftp.DownloadFile;

        using (Stream ftpStream = request.GetResponse().GetResponseStream())
        using (Stream fileStream = File.Create(@"C:\local\path\file.zip"))
        {
            byte[] buffer = new byte[10240];
            int read;
            while ((read = ftpStream.Read(buffer, 0, buffer.Length)) > 0)
            {
                fileStream.Write(buffer, 0, read);
                int position = (int)fileStream.Position;
                progressBar1.Invoke(
                    (MethodInvoker)(() => progressBar1.Value = position));
            }
        }
    }
    catch (Exception e)
    {
        MessageBox.Show(e.Message);
    }
}

enter image description here

ЯдроКод загрузки основан на:
Загрузка и загрузка двоичного файла на / с FTP-сервера в C # /. NET

0 голосов
/ 04 января 2011

FTP Не даст вам размеры контента, как HTTP, вам, вероятно, лучше сделать это самостоятельно.

FtpWebRequest FTPWbReq = WebRequest.Create("somefile") as FtpWebRequest;
FTPWbReq .Method = WebRequestMethods.Ftp.GetFileSize;

FtpWebResponse FTPWebRes = FTPWbReq.GetResponse() as FtpWebResponse;
long length = FTPWebRes.ContentLength;
FTPWebRes.Close();
...