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

Это модифицированный пример, который поставляется с libcurl. Я передаю bodyfile в функцию 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;
}

1 Ответ

2 голосов
/ 19 июня 2011

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

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

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

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

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

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

Страница man для curl_easy_setopt говорит следующее о CURLOPT_HEADERFUNCTION:

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

...