我仍在尝试通过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
答案 0 :(得分:0)
使用主机名填充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