Привязка клиента к серверу с использованием сокетов со случайным портом - PullRequest
1 голос
/ 18 июня 2019

Я пытаюсь создать чат-комнату между сервером и несколькими клиентами.Все полностью функционирует, кроме привязки сокета к случайному порту.Я создаю случайный порт через

server_info.sin_port = htons(0);

Моя проблема заключается в том, что клиент может использовать этот порт, или как я могу отправить номер порта клиенту?

Если я заменюпорт на статический номер, например '8888', программа успешно работает.У меня просто проблема с клиентом, собирающим порт.

Код сервера

int main()
{
    signal(SIGINT, catch_ctrl_c_and_exit);

    // Create socket
    server_sockfd = socket(AF_INET , SOCK_STREAM , 0);
    if (server_sockfd == -1) {
        char error[255];
        sprintf(error,"Fail to create a socket.");
    write(1,error,strlen(error));
        exit(EXIT_FAILURE);
    }

    // Socket information
    struct sockaddr_in server_info, client_info;
    int s_addrlen = sizeof(server_info);
    int c_addrlen = sizeof(client_info);
    memset(&server_info, 0, s_addrlen);
    memset(&client_info, 0, c_addrlen);
    server_info.sin_family = PF_INET;
    server_info.sin_addr.s_addr = INADDR_ANY;
    //TEST
    server_info.sin_port = htons(0);

    // Bind and Listen
    bind(server_sockfd, (struct sockaddr *)&server_info, s_addrlen);
    listen(server_sockfd, 5);

    // Print Server IP
    char start[100];
    getsockname(server_sockfd, (struct sockaddr*) &server_info, (socklen_t*) &s_addrlen);
    sprintf(start,"Start Server on: %s:%d\n", inet_ntoa(server_info.sin_addr), ntohs(server_info.sin_port));
    write(1,start,strlen(start));
    // Initial linked list for clients
    root = newNode(server_sockfd, inet_ntoa(server_info.sin_addr));
    now = root;

    while (1) {
        client_sockfd = accept(server_sockfd, (struct sockaddr*) &client_info, (socklen_t*) &c_addrlen);

        // Print Client IP
        getpeername(client_sockfd, (struct sockaddr*) &client_info, (socklen_t*) &c_addrlen);
    char client[255];
        sprintf(client,"Client %s:%d come in.\n", inet_ntoa(client_info.sin_addr), ntohs(client_info.sin_port));
    write(1,client,strlen(client));

        // Append linked list for clients
        ClientList *c = newNode(client_sockfd, inet_ntoa(client_info.sin_addr));
        c->prev = now;
        now->link = c;
        now = c;

        pthread_t id;
        if (pthread_create(&id, NULL, (void *)client_handler, (void *)c) != 0) {
            perror("Create pthread error!\n");
            exit(EXIT_FAILURE);
        }
    }

    return 0;
}

Код клиента

int main()
{
    signal(SIGINT, catch_ctrl_c_and_exit);

    // Naming
    char nick[255];
    sprintf(nick,"Please enter your name: ");
    write(1,nick,strlen(nick));
    if (fgets(nickname, LENGTH_NAME, stdin) != NULL) {
        str_trim_lf(nickname, LENGTH_NAME);
    }
    if (strlen(nickname) < 2 || strlen(nickname) >= LENGTH_NAME-1) {

      char nameerr[100];
      sprintf(nameerr,"\nName must be more than one and less than thirty characters.\n");
      write(1,nameerr,strlen(nameerr));
        exit(EXIT_FAILURE);
    }

    // Create socket
    sockfd = socket(AF_INET , SOCK_STREAM , 0);
    if (sockfd == -1) {
      char err[100];
      sprintf(err,"Fail to create a socket.");
      write(1,err,strlen(err));
        exit(EXIT_FAILURE);
    }

    // Socket information
    struct sockaddr_in server_info, client_info;
    int s_addrlen = sizeof(server_info);
    int c_addrlen = sizeof(client_info);
    memset(&server_info, 0, s_addrlen);
    memset(&client_info, 0, c_addrlen);
    server_info.sin_family = PF_INET;
    server_info.sin_addr.s_addr = inet_addr("127.0.0.1");
    //TEST
     server_info.sin_port = htons(0);

    // Connect to Server
    int err = connect(sockfd, (struct sockaddr *)&server_info, s_addrlen);
    if (err == -1) {
      char err[100];
      sprintf(err,"Connection to Server error!\n");
      write(1,err,strlen(err));
        exit(EXIT_FAILURE);
    }

    // Names
    getsockname(sockfd, (struct sockaddr*) &client_info, (socklen_t*) &c_addrlen);
    getpeername(sockfd, (struct sockaddr*) &server_info, (socklen_t*) &s_addrlen);
    char conn[100];
    char ipval[100];
    sprintf(conn,"Connect to Server: %s:%d\n", inet_ntoa(server_info.sin_addr), ntohs(server_info.sin_port));
    sprintf(ipval,"You are: %s:%d\n", inet_ntoa(client_info.sin_addr), ntohs(client_info.sin_port));
    write(1,conn,strlen(conn));
    write(1,ipval,strlen(ipval));

    send(sockfd, nickname, LENGTH_NAME, 0);

    pthread_t send_msg_thread;
    if (pthread_create(&send_msg_thread, NULL, (void *) send_msg_handler, NULL) != 0) {
      char err[100];
      sprintf (err,"Create pthread error!\n");
    write(1,err,strlen(err));
        exit(EXIT_FAILURE);
    }

    pthread_t recv_msg_thread;
    if (pthread_create(&recv_msg_thread, NULL, (void *) recv_msg_handler, NULL) != 0) {
      char err[100]; 
      sprintf(err,"Create pthread error!\n");
      write(1,err,strlen(err));
        exit(EXIT_FAILURE);
    }

    while (1) {
        if(flag) {
      char bye[20];
      sprintf(bye,"\nBye\n");
      write(1,bye,strlen(bye));
            break;
        }
    }

    close(sockfd);
    return 0;
}

Ошибка обнаружена

connect(sockfd, (struct sockaddr *)&server_info, s_addrlen);

1 Ответ

1 голос
/ 18 июня 2019

Вы можете отправить номер порта клиенту так же, как вы отправляете IP-адрес сервера.

Чтобы подключиться по TCP, клиент должен знать IP-адрес и ненулевое значениеномер порта сервера.

Если клиент и сервер работают на одном компьютере, обычно для передачи IP-адреса и номера порта обычно используются файлы, переменные среды или буфер обмена.

В вашем примере вы можете добавить printf в server.c, чтобы напечатать IP-адрес и номер порта (получить их с помощью функции getsockname ).Затем вы можете добавить код в client.c, чтобы получить IP-адрес от argv[1] и номер порта от argv[2].Закончив с изменениями кода выше, вы сначала запускаете сервер, копируете напечатанный IP-адрес и номер порта в буфер обмена и вставляете его в командную строку клиента, затем нажимаете Enter для запускаклиент, и он получит эти 2 параметра в своем argv.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...