C libcurl obtiene la salida en una cadena

6 minutos de lectura

avatar de usuario
frx08

Quiero almacenar el resultado de esta función curl en una variable, ¿cómo puedo hacerlo?

#include <stdio.h>
#include <curl/curl.h>

int main(void)
{
  CURL *curl;
  CURLcode res;

  curl = curl_easy_init();
  if(curl) {
    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    res = curl_easy_perform(curl);

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}

gracias lo solucione asi:

#include <stdio.h>
#include <stdlib.h>
#include <curl/curl.h>

function_pt(void *ptr, size_t size, size_t nmemb, void *stream){
    printf("%d", atoi(ptr));
}

int main(void)
{
  CURL *curl;
  curl = curl_easy_init();
  if(curl) {
    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_pt);
    curl_easy_perform(curl);
    curl_easy_cleanup(curl);
  }
  system("pause");
  return 0;
}

  • Solo para señalar en su solución en function_pt() que está convirtiendo la cadena en ptr en un número entero para convertirla nuevamente en una cadena en la salida. Puede generar la cadena directamente (y ver la respuesta completa).

    – zzz

    19 de junio de 2015 a las 3:33

  • aquí hay un enlace al ejemplo cURL curl.haxx.se/libcurl/c/getinmemory.html

    – lafferc

    30 de julio de 2015 a las 16:19

  • CURLcode res; no se usa

    – fnc12

    17 de septiembre de 2016 a las 11:14

  • la misma pregunta pero para C++ en lugar de c, vaya aquí: Guarde el resultado del contenido de cURL en una cadena en C++

    – Trevor Boyd Smith

    4 sep 2018 a las 14:36


avatar de usuario
alex jazmin

Puede configurar una función de devolución de llamada para recibir fragmentos de datos entrantes usando curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, myfunc);

La devolución de llamada tomará un argumento definido por el usuario que puede configurar usando curl_easy_setopt(curl, CURLOPT_WRITEDATA, p)

Aquí hay un fragmento de código que pasa un búfer struct string {*ptr; len} a la función de devolución de llamada y aumenta ese búfer en cada llamada usando realloc().

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <curl/curl.h>

struct string {
  char *ptr;
  size_t len;
};

void init_string(struct string *s) {
  s->len = 0;
  s->ptr = malloc(s->len+1);
  if (s->ptr == NULL) {
    fprintf(stderr, "malloc() failed\n");
    exit(EXIT_FAILURE);
  }
  s->ptr[0] = '\0';
}

size_t writefunc(void *ptr, size_t size, size_t nmemb, struct string *s)
{
  size_t new_len = s->len + size*nmemb;
  s->ptr = realloc(s->ptr, new_len+1);
  if (s->ptr == NULL) {
    fprintf(stderr, "realloc() failed\n");
    exit(EXIT_FAILURE);
  }
  memcpy(s->ptr+s->len, ptr, size*nmemb);
  s->ptr[new_len] = '\0';
  s->len = new_len;

  return size*nmemb;
}

int main(void)
{
  CURL *curl;
  CURLcode res;

  curl = curl_easy_init();
  if(curl) {
    struct string s;
    init_string(&s);

    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writefunc);
    curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);
    res = curl_easy_perform(curl);

    printf("%s\n", s.ptr);
    free(s.ptr);

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}

  • Agradable. Incluso mejor si todos esos size_t (además len mismo) sería declarado const.

    – alk

    08/06/2014 a las 18:31


  • para C++ std::string ven aquí

    – Trevor Boyd Smith

    04/09/2018 a las 14:40

avatar de usuario
El físico cuántico

La siguiente respuesta es la forma C++ de hacerlo, con std::string, en lugar de una cadena terminada en nulo. Todavía usa una función de devolución de llamada (no hay forma de evitarlo), pero también maneja el error de asignación usando try/catch.

#include <iostream>
#include <string>
#include <curl/curl.h>

size_t CurlWrite_CallbackFunc_StdString(void *contents, size_t size, size_t nmemb, std::string *s)
{
    size_t newLength = size*nmemb;
    try
    {
        s->append((char*)contents, newLength);
    }
    catch(std::bad_alloc &e)
    {
        //handle memory problem
        return 0;
    }
    return newLength;
}
int main()
{
    CURL *curl;
    CURLcode res;

    curl_global_init(CURL_GLOBAL_DEFAULT);

    curl = curl_easy_init();
    std::string s;
    if(curl)
    {

        curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");

        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); //only for https
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); //only for https
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, CurlWrite_CallbackFunc_StdString);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);
        curl_easy_setopt (curl, CURLOPT_VERBOSE, 1L); //remove this to disable verbose output


        /* Perform the request, res will get the return code */
        res = curl_easy_perform(curl);
        /* Check for errors */
        if(res != CURLE_OK)
        {
            fprintf(stderr, "curl_easy_perform() failed: %s\n",
                    curl_easy_strerror(res));
        }

        /* always cleanup */
        curl_easy_cleanup(curl);
    }

    std::cout<<s<<std::endl;

    std::cout<< "Program finished!" << std::endl;
}

  • Creo que std::string::append puede hacer que la función de devolución de llamada sea mucho más simple.

    –Ryan Burn

    5 de diciembre de 2018 a las 18:54

  • @rnickb Tienes razón; s->append((char*)contents. nmemb); funciona perfectamente conmigo y es más conciso. También el firma de función oficial para la devolución de llamada tiene un char* como primer argumento, por lo que podría usar eso y omitir la conversión. Y por último, s->resize() en realidad inicializa el espacio recién asignado. Como está a punto de sobrescribirlo de todos modos, s->reserve() sería más eficiente.

    – Jeinzi

    31 de enero de 2019 a las 14:53

  • Esto me ayudó mucho. ¿Puede también dar un ejemplo de cómo hacerlo con HTTP POST por favor 🙂

    – Señor Wolfenstein

    8 de enero de 2020 a las 15:34

De leer el manual aquí: http://curl.haxx.se/libcurl/c/curl_easy_setopt.html Creo que necesita varias llamadas a CURL_SETOPT, la primera es la URL que desea procesar, la segunda es algo como:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_ptr);

Donde function_ptr coincide con esta firma:

size_t function( void *ptr, size_t size, size_t nmemb, void *stream)

Lo que sucede aquí es que denota una función de devolución de llamada a la que libcurl llamará cuando tenga alguna salida para escribir desde cualquier transferencia que haya invocado. Puede hacer que escriba automáticamente en un archivo o pasarle un puntero a una función que manejará la salida por sí misma. Con esta función, debería poder ensamblar las diversas cadenas de salida en una sola pieza y luego usarlas en su programa.

No estoy seguro de qué otras opciones puede tener que configurar / qué más afecta cómo desea que se comporte su aplicación, así que eche un vistazo a esa página.

avatar de usuario
Pablo Grinberg

Aquí hay un sabor C ++ de la respuesta aceptada de alex-jasmin

#include <iostream>
#include <string>
#include <curl/curl.h>

size_t writefunc(void *ptr, size_t size, size_t nmemb, std::string *s) 
{
  s->append(static_cast<char *>(ptr), size*nmemb);
  return size*nmemb;
}

int main(void)
{
  CURL *curl = curl_easy_init();
  if (curl)
  {
    std::string s;

    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writefunc);
    curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);

    CURLcode res = curl_easy_perform(curl);

    std::cout << s << std::endl;

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}

¿Ha sido útil esta solución?

Esta web utiliza cookies propias y de terceros para su correcto funcionamiento y para fines analíticos y para mostrarte publicidad relacionada con sus preferencias en base a un perfil elaborado a partir de tus hábitos de navegación. Al hacer clic en el botón Aceptar, acepta el uso de estas tecnologías y el procesamiento de tus datos para estos propósitos. Configurar y más información
Privacidad