Как возобновить загрузку файлов через FTP с помощью Curl в C? - PullRequest
8 голосов
/ 23 февраля 2012

Я пытаюсь загрузить файл по FTP, и между ними, если соединение разорвано, оно должно возобновиться с того места, где оно было остановлено.Моя проблема заключается в том, что, используя следующий фрагмент кода, я могу продолжить загрузку, если я закрываю соединение, а затем снова подключаю его, но если я делаю это на сайте сервера, я не могу возобновить загрузку, и программа переходит в состояние бесконечности.

#include <stdio.h>

#include <curl/curl.h>

/*
 * This is an example showing how to get a single file from an FTP server.
 * It delays the actual destination file creation until the first write
 * callback so that it won't create an empty file in case the remote file
 * doesn't exist or something else fails.
 */

struct FtpFile {
  const char *filename;
  FILE *stream;
};

static size_t my_fwrite(void *buffer, size_t size, size_t nmemb, void *stream)
{
  struct FtpFile *out=(struct FtpFile *)stream;
  if(out && !out->stream) {
    /* open file for writing */
    out->stream=fopen(out->filename, "wb");
    if(!out->stream)
      return -1; /* failure, can't open file to write */
  }
  return fwrite(buffer, size, nmemb, out->stream);
}


int main(void)
{
  CURL *curl;
  CURLcode res;
  struct FtpFile ftpfile={
    "dev.zip", /* name to store the file as if succesful */
    NULL
  };

  curl_global_init(CURL_GLOBAL_DEFAULT);

  curl = curl_easy_init();
  if(curl) {
    /*
     * You better replace the URL with one that works!
     */
    curl_easy_setopt(curl, CURLOPT_URL,
                     "ftp://root:password@192.168.10.1/dev.zip");
    /* Define our callback to get called when there's data to be written */
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, my_fwrite);
    /* Set a pointer to our struct to pass to the callback */
    curl_easy_setopt(curl, CURLOPT_WRITEDATA, &ftpfile);

    /* Switch on full protocol/debug output */
    curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);

    res = curl_easy_perform(curl);

    /* always cleanup */
    curl_easy_cleanup(curl);

    if(CURLE_OK != res) {
      /* we failed */
      fprintf(stderr, "curl told us %d\n", res);
    }
  }

  if(ftpfile.stream)
    fclose(ftpfile.stream); /* close the local file */

  curl_global_cleanup();

  return 0;
}

Может кто-нибудь сказать мне, как я могу возобновить загрузку, если соединение закрыто удаленным сайтом.Любая помощь будет оценена

Спасибо,

Yuvi

Ответы [ 2 ]

2 голосов
/ 23 февраля 2012

Добавьте переменную в структуру ftpfile, чтобы понять, что вашей функции записи необходимо добавить и сказать libcurl возобновить загрузку с конца файла назначения, установив для CURLOPT_RESUME_FROM количество загруженных байтов:

 struct FtpFile 
 { 
   const char *pcInfFil; 
   FILE *pFd; 
   int iAppend; 
 };

В основном, если вы хотите возобновить:

curl_easy_setopt(curl, CURLOPT_RESUME_FROM , numberOfBytesToSkip); 

Если файл еще не существует или это не возобновленная загрузка, а новая загрузка, обязательно установите для CURLOPT_RESUME_FROM значение 0.

В my_fwrite:

out->stream=fopen(out->filename, out->iAppend ? "ab":"wb"); 

P.S. если вам нужно возобновить файл больше, чем long (2 ГБ), посмотрите CURLOPT_RESUME_FROM_LARGE и CURL_OFF_T_C ()

В ответ на комментарий, запрашивающий дополнительную информацию о том, как узнать, когда передача не удалась:

После того, как вы наберете curl, просто выполните вызов:

CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ... );

Извлечение из контекста curl:

CURLINFO_HEADER_SIZE 
CURLINFO_CONTENT_LENGTH_DOWNLOAD

Сложите их вместе и убедитесь, что они равны

CURLINFO_SIZE_DOWNLOAD

Если нет, попытайтесь повторить контекст.

И обязательно используйте последнюю версию curl, она должна прерваться через 60 секунд после того, как не будет слышен FTP-сервер, с которого она загружается.

0 голосов
/ 16 марта 2012

Вы можете использовать параметры CURLOPT_CONNECTTIMEOUT и CURLOPT_TIMEOUT, чтобы указать время ожидания подключения и максимальное время выполнения для каждого дескриптора.

Другой способ (который работает только в том случае, если вы используете простой интерфейс, а не мультиинтерфейс) - это использовать обратный вызов опции сокета, который вы можете установить с помощью CURLOPT_SOCKOPTFUNCTION. В нем вы должны вызывать setsockopt () для параметра SO_RCVTIMEO до максимального времени, в течение которого соединение может простаивать, прежде чем оно должно быть сброшено. то есть, если за последние 5 секунд не было получено ни одного байта, прервите соединение.

...