Libcurl: почему curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, obj) передает мой указатель на функцию записи, который не равен obj?

Это модифицированный пример, который поставляется с libcurl. Я передаю файл тела в функцию write_data, устанавливая CURLOPT_WRITEDATA. Он действительно записывает данные в файл, но указатель не равен указателю, переданному в setopt(curl_handle, CURLOPT_WRITEDATA, указатель). И я получаю SIGSEGV, если пытаюсь передать указатель на объект, отличный от FILE. Почему? Как я могу передать тот же указатель?

Код:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <assert.h>

#include <curl/curl.h>
#include <curl/types.h>
#include <curl/easy.h>
FILE *bodyfile;
static size_t write_data(void *ptr, size_t size, size_t nmemb, void *stream)
{
    assert(bodyfile == (FILE*) stream); //this assertion fails, but when i comment it, code works. Why?
    int written = fwrite(ptr, size, nmemb, (FILE *)stream);
    return written;
}

int main(void)
{
    CURL *curl_handle;
    static const char *headerfilename = "head.out";
    FILE *headerfile;
    static const char *bodyfilename = "body.out";

    curl_global_init(CURL_GLOBAL_ALL);

    /* init the curl session */
    curl_handle = curl_easy_init();

    /* set URL to get */
    curl_easy_setopt(curl_handle, CURLOPT_URL, "http://google.com");

    /* no progress meter please */
    curl_easy_setopt(curl_handle, CURLOPT_NOPROGRESS, 1L);

    /* send all data to this function  */
    curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, write_data);

    /* open the files */
    headerfile = fopen(headerfilename,"w");
    if (headerfile == NULL) {
        curl_easy_cleanup(curl_handle);
        return -1;
    }
    bodyfile = fopen(bodyfilename,"w");
    if (bodyfile == NULL) {
        curl_easy_cleanup(curl_handle);
        return -1;
    }

    /* we want the headers to this file handle */
    curl_easy_setopt(curl_handle,   CURLOPT_WRITEHEADER, headerfile);

    /* we want the body to this file handle */
    curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, bodyfile);

    /* get it! */
    curl_easy_perform(curl_handle);

    /* close the header file */
    fclose(headerfile);

    /* cleanup curl stuff */
    curl_easy_cleanup(curl_handle);

    return 0;
}

person frp    schedule 19.06.2011    source источник


Ответы (1)


Если вы удалите утверждение, вставьте:

printf("body: %p stream: %p\n", bodyfile, stream);

в write_data, а затем добавьте:

printf("head: %p body: %p\n", headerfile, bodyfile);

после открытия и шапки, и тела файла все станет ясно.

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

На странице руководства для curl_easy_setopt сказано следующее о CURLOPT_HEADERFUNCTION:

Если этот параметр не установлен или если он имеет значение NULL, но CURLOPT_HEADERDATA (CURLOPT_WRITEHEADER) имеет любое значение, кроме NULL, вместо него будет использоваться функция, используемая для приема данных ответа. То есть это будет функция, указанная с помощью CURLOPT_WRITEFUNCTION, или, если она не указана или NULL, — функция по умолчанию, записывающая поток.

person Puppe    schedule 19.06.2011