solicitud http usando sockets en c ++

solicitud http usando sockets en c ++

Estoy tratando de hacer una solicitud HTTP usando sockets en Linux, y mi función funciona ahora, pero solo con dominios simples como www.google.com o webpage.000webhostapp.com. Cuando intento agregar una ruta y una consulta como webpage.000webhostapp.com/folder/file.php?parameter=value, comienza a fallar.

Este es mi código:

#include <iostream>
#include <cstring>
#include <string>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netdb.h>
#include <stdio.h>

int main(){
 int socket_desc;
 struct sockaddr_in serv_addr;
 struct hostent *server;
 char buffer[4096];

 std::string url = "www.exampleweb.com/folder/file.php";

 socket_desc = socket(AF_INET, SOCK_STREAM, 0);
 if(socket_desc < 0){
  std::cout<<"failed to create socket"<<std::endl;
 }

 server = gethostbyname(url.c_str());
 if(server==NULL){std::cout<<"could Not resolve hostname :("<<std::endl;}
 bzero((char *) &serv_addr, sizeof(serv_addr));
 serv_addr.sin_family = AF_INET;
 serv_addr.sin_port = htons(80);
 bcopy((char *)server->h_addr,
         (char *)&serv_addr.sin_addr.s_addr,
         server->h_length);

 if(connect(socket_desc, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0){
   std::cout<<"connection failed :("<<std::endl;
 }
 std::string request = "GET / HTTP/1.1\r\nHost: " + url + "\r\nConnection: close\r\n\r\n";

 if(send(socket_desc, request.c_str(), strlen(request.c_str())+1, 0) < 0){
  std::cout<<"failed to send request..."<<std::endl;
 }
 int n;
 std::string raw_site;
 while((n = recv(socket_desc, buffer, sizeof(buffer)+1, 0)) > 0){
  int i = 0;
        while (buffer[i] >= 32 || buffer[i] == '\n' || buffer[i] == '\r'){

            raw_site+=buffer[i];
            i += 1;
        }
 }

 std::cout<<raw_site<<std::endl;
 return 0;
}

Usando www.google.com, funciona perfectamente para la solicitud.

Usando www.example.com/folder/file.php?parameter=value&parameter2=value2, falla y devuelve "could not resolve hostname".

¿Alguna idea de cómo solucionarlo?

No estoy usando curl y no puedo usarlo para este proyecto.

Mostrar la mejor respuesta

Debe analizar la URL para obtener sus partes. O use libcurl.

También aprenda el protocolo http con más cuidado. GET / HTTP/1.1 no es válido para la URL "www.exampleweb.com/carpeta/archivo.php".

gethostbyname() (que por cierto está obsoleto, debería usar getaddrinfo() en su lugar) no funciona con URL, solo con nombres de host.

Dada una URL como http://www.exampleweb.com/folder/file.php?parameter=value&parameter2=value2, primero debe dividirla en sus partes constituyentes (lea RFC 3986), por ejemplo:

  • el esquema http
  • el nombre de host www.exampleweb.com
  • la ruta del recurso /folder/file.php
  • la consulta ?parameter=value&parameter2=value2

Luego, puede resolver la dirección IP del host, conectarse a esa dirección IP en el puerto 80 (el puerto predeterminado para HTTP), ya que la URL no especifica un puerto diferente, y finalmente enviar un GET solicitud del recurso y consulta.

Además, tenga en cuenta que el encabezado Host en la solicitud GET debe especificar solo el nombre del host (y el puerto, si es diferente al predeterminado), no toda la URL. Y también que las solicitudes HTTP no son cadenas terminadas en nulo, por lo que no envíe el terminador nulo. Lea RFC 2616.

Prueba esto en su lugar:

#include <iostream>
#include <cstring>
#include <string>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netdb.h>
#include <stdio.h>

int main(){
    int socket_desc;
    struct sockaddr_in serv_addr;
    struct hostent *server;
    char buffer[4096];

    std::string host = "www.exampleweb.com";
    std::string port = "80";
    std::string resource = "/folder/file.php";
    std::string query = "?parameter=value&parameter2=value2";

    socket_desc = socket(AF_INET, SOCK_STREAM, 0);
    if (socket_desc < 0){
        std::cout << "failed to create socket" << std::endl;
        return 0;
    }

    server = gethostbyname(host.c_str());
    if (server == NULL){
        std::cout << "could Not resolve hostname :(" << std::endl;
        close(socket_desc);
        return 0;
    }

    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(std::stoi(port));
    bcopy((char *)server->h_addr, (char *)&serv_addr.sin_addr.s_addr, server->h_length);

    if (connect(socket_desc, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0){
        std::cout << "connection failed :(" << std::endl;
        close(socket_desc);
        return 0;
    }

    std::string request = "GET " + resource + query + " HTTP/1.1\r\nHost: " + host + "\r\nConnection: close\r\n\r\n";

    if (send(socket_desc, request.c_str(), request.size(), 0) < 0){
        std::cout << "failed to send request..." << std::endl;
        close(socket_desc);
        return 0;
    }

    int n;
    std::string raw_site;
    while ((n = recv(socket_desc, buffer, sizeof(buffer), 0)) > 0){
        raw_site.append(buffer, n);
    }

    close(socket_desc);

    std::cout << raw_site << std::endl;
    return 0;
}