【问题标题】:http request using sockets on c++在 C++ 上使用套接字的 http 请求
【发布时间】:2022-01-01 18:33:19
【问题描述】:

我正在尝试在 Linux 上使用套接字发出 HTTP 请求,我的功能现在可以工作,但仅适用于 www.google.comwebpage.000webhostapp.com 等简单域。当我尝试添加像webpage.000webhostapp.com/folder/file.php?parameter=value 这样的路径和查询时,它开始失败。

这是我的代码:

#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;
}

使用www.google.com,它非常适合请求。

使用www.example.com/folder/file.php?parameter=value&amp;parameter2=value2,失败并返回"could not resolve hostname"

知道如何解决它吗?

我没有使用 curl,也不能在这个项目中使用它。

【问题讨论】:

  • 您应该解析 url 以获取其部分。或者使用 libcurl。
  • 还要更仔细地学习http协议。 GET / HTTP/1.1 对网址“www.exampleweb.com/folder/file.php”无效。

标签: c++ http sockets request


【解决方案1】:

gethostbyname()(BTW 已弃用,您应该改用 getaddrinfo())不适用于 URL,只能用于主机名。

给定一个像 http://www.exampleweb.com/folder/file.php?parameter=value&amp;parameter2=value2 这样的 URL,你需要先把它分解成它的组成部分(阅读 RFC 3986),例如:

  • 方案http
  • 主机名www.exampleweb.com
  • 资源路径/folder/file.php
  • 查询?parameter=value&amp;parameter2=value2

然后您可以解析主机的 IP 地址,在端口 80(HTTP 的默认端口)上连接到该 IP 地址,因为 URL 没有指定不同的端口,最后发送 GET 请求资源和查询。

另外,请注意GET 请求中的Host 标头必须指定主机名(和端口,如果与默认值不同),而不是整个 URL。而且 HTTP 请求不是以 null 结尾的字符串,所以不要发送 null 终止符。阅读RFC 2616

试试这个:

#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;
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2015-12-29
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-05-14
    • 2014-02-21
    相关资源
    最近更新 更多