C++ Socks5 通信问题

Posted

技术标签:

【中文标题】C++ Socks5 通信问题【英文标题】:C++ Socks5 communication Issues 【发布时间】:2014-05-25 20:34:39 【问题描述】:

我仍在尝试通过 SOCKS5 代理连接到服务器。我已阅读 RFC1928,现在我正在使用此代码,但它没有按预期工作。 发送域名似乎有问题。当将 addr 位更改为 1 并使用 IP 地址时,它工作得很好。

我使用的代码是这样的:

#define PUT_BYTE(ptr,data) (*(unsigned char*)ptr = data)
struct sockaddr_in saddr;
saddr.sin_port = htons(9150);
saddr.sin_family = AF_INET;
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
int rv = connect(fd, (struct sockaddr *)&saddr, sizeof(saddr));
if (rv < SOCKET_ERROR)
    return 1;

char buf[256], *ptr;
ptr = buf;
PUT_BYTE(ptr++, 5); // Version 5
PUT_BYTE(ptr++, 1);
PUT_BYTE(ptr++, 0x00); // No Auth
send(fd, buf, ptr - buf, 0);
recv(fd, buf, 2, 0);
if ((buf[0] != 5) || buf[1] == 0xFF)

    exit(0);

ptr = buf;
PUT_BYTE(ptr++, 5); // Version 5
PUT_BYTE(ptr++, 1); // Connect
PUT_BYTE(ptr++, 0); // Reserved
PUT_BYTE(ptr++, 3); // Use Hostname
//memcpy(ptr, &destaddr.sin_addr.s_addr, sizeof(destaddr.sin_addr));
//ptr += sizeof(destaddr.sin_addr);
//PUT_BYTE(ptr++, static_cast<unsigned char>(21));
PUT_BYTE(ptr++, 22); // Set Hostname lenght
//ptr += static_cast<unsigned char>(22);
//ptr += static_cast<unsigned char>(address.c_str());
//memcpy(ptr, &"iszgnywrejvdg2nc.onion", sizeof("iszgnywrejvdg2nc.onion") + 4);
PUT_BYTE(ptr++, (byte)"iszgnywrejvdg2nc.onion"); // set Hostname
PUT_BYTE(ptr++, 0xFF); //Set Hostname end
PUT_BYTE(ptr++, dest_port >> 8); //set port
PUT_BYTE(ptr++, dest_port & 0xFF); //set port end
send(fd, buf, ptr - buf, 0); //send out data
recv(fd, buf, 4, 0); // read response
if (!buf[1] != 0x00) // check response

    ptr = buf + 4;
    switch (buf[3]) 
    case 1:
        recv(fd, ptr, 4 + 2, 0);
        break;
    case 3:
        recv(fd, ptr, 1, 0);
        recv(fd, ptr + 1, *(unsigned char*)ptr + 2, 0);
        break;
    case 4:
        recv(fd, ptr, 16 + 2, 0);
        break;
    

这是应该与上述匹配的 RFC

    +----+-----+-------+------+----------+----------+
    |VER | CMD |  RSV  | ATYP | DST.ADDR | DST.PORT |
    +----+-----+-------+------+----------+----------+
    | 1  |  1  | X'00' |  1   | Variable |    2     |
    +----+-----+-------+------+----------+----------+

 Where:

      o  VER    protocol version: X'05'
      o  CMD
         o  CONNECT X'01'
         o  BIND X'02'
         o  UDP ASSOCIATE X'03'
      o  RSV    RESERVED
      o  ATYP   address type of following address
         o  IP V4 address: X'01'
         o  DOMAINNAME: X'03'
         o  IP V6 address: X'04'
      o  DST.ADDR       desired destination address
      o  DST.PORT desired destination port in network octet
         order

【问题讨论】:

你能评论一下你的代码吗?我们不必查找文档参考指南来遵循您的代码。 @S.Richmond 我已经评论并包含了相应的 RFC 我建议您使用 Wireshark 之类的工具来检查您正在创建和发送的数据包,以确保它们的构造正确。 @S.Richmond Wireshark 将此 05 01 00 03 16 34 ff 4b fc 显示为第二个发送请求。 16 之后的一切都是意料之外的。有人可以解释那里发生了什么吗?? 【参考方案1】:

您使用主机名填充dst.addr 字段的代码完全错误。特别是在这一行:

PUT_BYTE(ptr++, (byte)"iszgnywrejvdg2nc.onion"); // set Hostname

您正在将char* 指针类型转换为单个字节。您正在存储截断指针值的最后一个字节,您根本没有将主机名数据复制到该字段中。您应该改用memcpy()strcpy(),这与您在注释代码中所做的类似(但也有错误)。

另外,这一行根本不属于,需要删除:

PUT_BYTE(ptr++, 0xFF); //Set Hostname end

试试类似的方法:

char buf[263], *ptr;
...
ptr = buf;
PUT_BYTE(ptr++, 5); // Version 5
PUT_BYTE(ptr++, 1); // Connect
PUT_BYTE(ptr++, 0); // Reserved
PUT_BYTE(ptr++, 3); // Use Hostname
char *hostname = "iszgnywrejvdg2nc.onion";
unsigned char len = (unsigned char) strlen(hostname);
PUT_BYTE(ptr++, len); // Set Hostname length
memcpy(ptr, hostname, len); // set Hostname
ptr += len;
PUT_BYTE(ptr++, dest_port >> 8); //set port
PUT_BYTE(ptr++, dest_port & 0xFF); //set port end

【讨论】:

以上是关于C++ Socks5 通信问题的主要内容,如果未能解决你的问题,请参考以下文章

如何正确认识socks5代理ip

C++ 和 Java 对象通信

进程间通信的问题(C++高手进)

没有 SslStream 的 SSL,因为我想通过 SOCKS5 代理连接

解决通信计时问题(C++ windows 应用程序)

C++ 跨进程通信