Nano Hash - криптовалюты, майнинг, программирование

Как назначить имя хоста и номер порта, с которого подключался клиент?

У меня есть эта программа ftserver.c, которая реализует сервер передачи файлов, который прослушивает клиента, а затем отвечает на запрос клиента через соединение для передачи данных. Сейчас это работает, но у меня жестко запрограммировано имя хоста и номер порта для подключения к данным. Номер порта был предоставлен клиентом, и сервер должен иметь возможность получить имя хоста из управляющего соединения клиента.

Ссылки:http://beej.us/guide/bgnet/output/html/multipage/getaddrinfoman.html

Как я могу динамически назначить имя хоста и номер порта? Спасибо.

void error(const char *msg)
{
perror(msg);
exit(1);
}

void startup(int portNumber);
void setupData(char* portNum);

int sockfd, newsockfd, datasock, portno;
char buffer[256]; socklen_t clilen;
struct sockaddr_in serv_addr, cli_addr, port_addr;
struct addrinfo hints, *servinfo, *p;
char ipstr[1000];
struct in_addr ipAddr;
struct sockaddr_in *s;

int main(int argc, char *argv[]){
    int n; char* dataport; char * token; char filename[100];

    if (argc < 2) {
        fprintf(stderr,"ERROR, no port provided\n");
        exit(1);
    }
    portno = atoi(argv[1]);

    startup(portno);

    n = read(newsockfd,buffer,255);
    if (n < 0) error("ERROR reading from socket");
    printf("Here is the message: %s\n",buffer);

    token = strtok(buffer, " ");

    //if client requested a list, setup data connection and send it
    if (strcmp(token, "-l") == 0){
      token = strtok(NULL, " ");
      printf("the token is %s\n", token);
      //dataport = atoi(token);
      dataport = token;
      setupData(dataport);
      //sendList(dataport);
    }
    //if client requested a file, setup data connection and send it
    else if (strcmp(token, "-g") == 0){
      token = strtok(NULL, " ");
      //filename = *token;
      token = strtok(NULL, " ");
      //dataport = atoi(token);
      printf("the data port is %d\n", dataport);
      //setupData(dataport);
      //sendFile(filename, dataport);
    }
    else {
      n = write(newsockfd,"not a valid command",19);
      if (n < 0) error("ERROR writing to socket");
    }

    //close sockets for connection P
    close(datasock);
    close(newsockfd);
    close(sockfd);
    return 0; 
}

void startup(int portNumber)
{

 sockfd = socket(AF_INET, SOCK_STREAM, 0);
 if (sockfd < 0) 
    error("ERROR opening socket");
 bzero((char *) &serv_addr, sizeof(serv_addr));

 serv_addr.sin_family = AF_INET;
 serv_addr.sin_addr.s_addr = INADDR_ANY;
 serv_addr.sin_port = htons(portNumber);
 if (bind(sockfd, (struct sockaddr *) &serv_addr,
          sizeof(serv_addr)) < 0) 
          error("ERROR on binding");
 listen(sockfd,5);
 clilen = sizeof(cli_addr);
 newsockfd = accept(sockfd, 
             (struct sockaddr *) &cli_addr, 
             &clilen);

 if (newsockfd < 0) 
      error("ERROR on accept");
 bzero(buffer,256);
}

void setupData(char* portNum){
  int rv;

  const char* name = "localhost";
  char s[1000];

  memset(&hints, 0, sizeof (hints));
  hints.ai_family = AF_UNSPEC; // use AF_INET6 to force IPv6
  hints.ai_socktype = SOCK_STREAM;

  if ((rv = getaddrinfo(name, "30024", &hints, &servinfo)) != 0) {
        fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv));
        exit(1);
  }

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

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

        break; // if we get here, we must have connected successfully
  }

  if (p == NULL) {
        // looped off the end of the list with no connection
        fprintf(stderr, "failed to connect\n");
        exit(2);
    }

  printf("data connection setup successful\n");
}
13.03.2017

  • Прямо сейчас, если я просто введу portNum вместо 30024 для getaddrinfo, это даст имя сервера, не поддерживаемое для ошибки ai_socktype. 14.03.2017
  • Почему вы звоните getaddrinfo(), если у вас уже есть имя хоста и порт клиента? 14.03.2017
  • Прямо сейчас я жестко запрограммировал его как localhost и 30024, но я хотел бы сделать его динамическим, чтобы параметры для моего отдельного клиентского кода могли указывать порт данных для запуска подключения данных и отправки информации на этот порт. Я попытался сохранить имя хоста клиента из первого управляющего соединения (созданного при запуске()), а затем применить его ко второму соединению, но не смог заставить его работать. 14.03.2017
  • Я решил это. Используя упрощенную версию настройки соединения данных, не используя getaddrinfo() и используя функцию преобразования hostname_to_ip. Ссылка: linuxhowtos.org/data/6/client.c 14.03.2017
  • Вам даже не нужно этого делать - IP-адрес будет доступен из клиентского сокета. 15.03.2017
  • для удобства чтения и понимания: 1) последовательно делать отступы в коде. 15.03.2017
  • спрашивая о проблеме во время выполнения, среди прочего, опубликуйте код, который компилируется чисто. В опубликованном коде, среди прочего, отсутствуют необходимые операторы #include. Вы ожидаете, что мы догадаемся, какие заголовочные файлы вы включили? 15.03.2017

Ответы:


Новые материалы

Кластеризация: более глубокий взгляд
Кластеризация — это метод обучения без учителя, в котором мы пытаемся найти группы в наборе данных на основе некоторых известных или неизвестных свойств, которые могут существовать. Независимо от..

Как написать эффективное резюме
Предложения по дизайну и макету, чтобы представить себя профессионально Вам не позвонили на собеседование после того, как вы несколько раз подали заявку на работу своей мечты? У вас может..

Частный метод Python: улучшение инкапсуляции и безопасности
Введение Python — универсальный и мощный язык программирования, известный своей простотой и удобством использования. Одной из ключевых особенностей, отличающих Python от других языков, является..

Как я автоматизирую тестирование с помощью Jest
Шутка для победы, когда дело касается автоматизации тестирования Одной очень важной частью разработки программного обеспечения является автоматизация тестирования, поскольку она создает..

Работа с векторными символическими архитектурами, часть 4 (искусственный интеллект)
Hyperseed: неконтролируемое обучение с векторными символическими архитектурами (arXiv) Автор: Евгений Осипов , Сачин Кахавала , Диланта Хапутантри , Тимал Кемпития , Дасвин Де Сильва ,..

Понимание расстояния Вассерштейна: мощная метрика в машинном обучении
В обширной области машинного обучения часто возникает необходимость сравнивать и измерять различия между распределениями вероятностей. Традиционные метрики расстояния, такие как евклидово..

Обеспечение масштабируемости LLM: облачный анализ с помощью AWS Fargate и Copilot
В динамичной области искусственного интеллекта все большее распространение получают модели больших языков (LLM). Они жизненно важны для различных приложений, таких как интеллектуальные..