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 通信问题的主要内容,如果未能解决你的问题,请参考以下文章