【问题标题】:Socks 5 client in CC语言中的Socks 5客户端
【发布时间】:2016-05-30 09:53:28
【问题描述】:

我想使用 C 中的流套接字编写 socks 5 客户端。

这是我的代码:

/*
** client_socks.c -- a stream socket socks 5 client demo
*/

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

#include <arpa/inet.h>

#define PORT "9050" // the port client will be connecting to 

#define MAXDATASIZE 100 // max number of bytes we can get at once 

// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa)
{
    if (sa->sa_family == AF_INET) {
        return &(((struct sockaddr_in*)sa)->sin_addr);
    }

    return &(((struct sockaddr_in6*)sa)->sin6_addr);
}

int main(int argc, char *argv[])
{
    int sockfd, numbytes;  
    char buf[MAXDATASIZE];
    struct addrinfo hints, *servinfo, *p;
    int rv;
    char s[INET6_ADDRSTRLEN];

    if (argc != 2) {
        fprintf(stderr,"usage: client hostname\n");
        exit(1);
    }

    memset(&hints, 0, sizeof hints);
    hints.ai_family = AF_UNSPEC;
    hints.ai_socktype = SOCK_STREAM;

    if ((rv = getaddrinfo(argv[1], PORT, &hints, &servinfo)) != 0) {
        fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
        return 1;
    }

    // loop through all the results and connect to the first we can
    for(p = servinfo; p != NULL; p = p->ai_next) {
        if ((sockfd = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) {
            perror("client: socket");
            continue;
        }

        if (connect(sockfd, p->ai_addr, p->ai_addrlen) == -1) {
            perror("client: connect");
            close(sockfd);
            continue;
        }

        break;
    }

    if (p == NULL) {
        fprintf(stderr, "client: failed to connect\n");
        return 2;
    }

    inet_ntop(p->ai_family, get_in_addr((struct sockaddr *)p->ai_addr), s, sizeof s);
    printf("client: connecting to %s\n", s);

    freeaddrinfo(servinfo); // all done with this structure

    /* Version 5, one method: no authentication */
    char buffer[256], *ptrBuff;
    ptrBuff = buffer;

    ptrBuff[0] = 5; // socks version
    ptrBuff[1] = 1;
    ptrBuff[2] = 0; // SOCKS_NOAUTH

    if (send(sockfd, ptrBuff, buffer-ptrBuff, 0) == -1)
                perror("send");

    if ((numbytes = recv(sockfd, buffer, MAXDATASIZE-1, 0)) == -1) {
        perror("recv");
        exit(1);
    }

    buffer[numbytes] = '\0';

    printf("client: received '%s'\n",buffer);

    close(sockfd);

    return 0;
}

但它不起作用 - 它正在写入:

client: connecting to 127.0.0.1 

等待,仅此而已

我还没有找到如何在 C 中执行此操作的真正有用的示例,因此我非常感谢您能给我的任何帮助。

如何实现auth和no auth方法?

新的更正代码:

/*
** client_proxy.c -- a stream socket socks 5 client demo
*/

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>

#include <arpa/inet.h>

#define PORT "9050" // the port client will be connecting to 

#define MAXDATASIZE 100 // max number of bytes we can get at once 

// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa)
{
    if (sa->sa_family == AF_INET) {
        return &(((struct sockaddr_in*)sa)->sin_addr);
    }

    return &(((struct sockaddr_in6*)sa)->sin6_addr);
}

int main(int argc, char *argv[])
{
    int sockfd, numbytes;  
    char buf[MAXDATASIZE];
    struct addrinfo hints, *servinfo, *p;
    int rv;
    char s[INET6_ADDRSTRLEN];

    if (argc != 2) {
        fprintf(stderr,"usage: client hostname\n");
        exit(1);
    }

    memset(&hints, 0, sizeof hints);
    hints.ai_family = AF_UNSPEC;
    hints.ai_socktype = SOCK_STREAM;

    if ((rv = getaddrinfo(argv[1], PORT, &hints, &servinfo)) != 0) {
        fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
        return 1;
    }

    // loop through all the results and connect to the first we can
    for(p = servinfo; p != NULL; p = p->ai_next) {
        if ((sockfd = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) {
            perror("client: socket");
            continue;
        }

        if (connect(sockfd, p->ai_addr, p->ai_addrlen) == -1) {
            perror("client: connect");
            close(sockfd);
            continue;
        }

        break;
    }

    if (p == NULL) {
        fprintf(stderr, "client: failed to connect\n");
        return 2;
    }

    inet_ntop(p->ai_family, get_in_addr((struct sockaddr *)p->ai_addr), s, sizeof s);
    printf("client: connecting to %s\n", s);

    freeaddrinfo(servinfo); // all done with this structure

    /* Version 5, one method: no authentication */
    char buffer[256], *ptrBuff, bufferf[256];
    ptrBuff = buffer;

    *(ptrBuff++) = 5; // socks version
    *(ptrBuff++) = 2;
    *(ptrBuff++) = 0x00; // no auth 
    *(ptrBuff++) = 0x02; // user pass auth

    if (send(sockfd, ptrBuff, buffer-ptrBuff, 0) == -1)
                perror("send");

    printf("sent\n");

    if ((numbytes = recv(sockfd, bufferf, MAXDATASIZE-1, 0)) == -1) {
        perror("recv");
        exit(1);
    }

    printf("client: received '%d'\n", bufferf[1]);

    close(sockfd);

    return 0;
}

我明白了:

client: connecting to 127.0.0.1
sent
client: received '97

每次收到的号码都不一样,怎么回事?

【问题讨论】:

  • 也许你应该比“它不工作”更具体一点......
  • 确实不行,否则你可能不会在这里。
  • 正在写入客户端:连接到 127.0.0.1 并等待,仅此而已
  • 使用你的调试器和更多的中间变量,例如。 'int numBytesToSend=buffer-ptrBuff;'

标签: c linux sockets socks


【解决方案1】:

深入了解这段代码:

char buffer[256], *ptrBuff;
ptrBuff = buffer;

ptrBuff[0] = 5; // socks version
ptrBuff[1] = 1;
ptrBuff[2] = 0; // SOCKS_NOAUTH

if (send(sockfd, ptrBuff, buffer-ptrBuff, 0) == -1)
            perror("send");

您正在将len:buffer-ptrBuff 传递给发送函数。但是这些值(地址)是相同的地址,所以结果是0

我猜,你想做的是

char buffer[256], *ptrBuff;
ptrBuff = buffer;

*(ptrBuff++) = 5; // socks version
*(ptrBuff++) = 1;
*(ptrBuff++) = 0; // SOCKS_NOAUTH

if (send(sockfd, ptrBuff, buffer-ptrBuff, 0) == -1)
            perror("send");

在这种情况下,ptrBuff 值(地址指向)在每次分配时递增,因此当调用 send 时,len (buffer-ptrBuff) 将是 3

最后,您看到的是正确的,因为您没有向服务器端发送任何内容,并且客户端无法收到任何答复。所以recv 永远等待(阻塞模式)一个永远不会回答的答案。

【讨论】:

  • 感谢指正,请看我的进一步结果
  • 不要编辑您的原始问题。 您必须按已回答的方式关闭此问题(选择一个正确的答案)并发布另一个包含新问题的单独问题。
【解决方案2】:
ptrBuff = buffer;
..
 //stuff that does not modify ptrBuff
..
send(sockfd, ptrBuff, buffer-ptrBuff, 0) // ie. send nothing.

【讨论】:

  • 谢谢,请看一下我的进一步结果
【解决方案3】:

这个问题没有完全回答。

根据 RFC (https://datatracker.ietf.org/doc/html/rfc1928),VER 和 NMETHODS 是十六进制值:

*(ptrBuff++) = 0x05; // socks version
*(ptrBuff++) = 0x02;

您真正的错误是缓冲区指向数据的开头,而ptrBuff 指向数据的结尾(因此,buffer-ptrBuff 是负值)。 那会更好:

if (send(sockfd, buffer, ptrBuff-buffer, 0) == -1)
            perror("send");

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2014-12-28
    • 2011-02-19
    • 2021-08-23
    • 2021-05-09
    • 2015-05-06
    • 1970-01-01
    • 2019-12-20
    相关资源
    最近更新 更多