Как получить мой IP-адрес без обратной связи в C?
-
05-07-2019 - |
Вопрос
Для связи между двумя хостами мне нужно отправить IP-адрес моего хоста на другой сайт. Проблема заключается в том, что если я запрашиваю свой IP-адрес, возможно, я получаю обратно свои локальные петлевые IP-адреса (127.x.x.x), а не сетевой (ethernet) IP-адрес.
Я использую следующий код:
char myhostname[32];
gethostname(myhostname, 32);
hp = gethostbyname(myhostname);
unsigned my_ip = *(unsigned*)(hp->h_addr);
if( (my_ip % 256) == 127) {
/* Wrong IP adress as it's 127.x.x.x */
printf("Error, local IP address!");
return;
}
Единственный способ решить эту проблему - убедиться, что мое имя хоста в / etc / hosts находится за реальным сетевым адресом, а не за локальной обратной связью (по умолчанию, например, в Ubuntu).
Есть ли способ решить эту проблему, не полагаясь на содержимое / etc / hosts?
Правка . Я изменил приведенный выше код, чтобы он использовал getaddrinfo, но я все равно получаю номер устройства обратной связи (127.0,0,1) вместо реального IP-адреса:
struct addrinfo hint = {0};
struct addrinfo *aip = NULL;
unsigned ip = 0;
struct sockaddr_in *sinp = NULL;
hint.ai_family = AF_INET; /* IPv4 */
hint.ai_socktype = SOCK_STREAM;
if(getaddrinfo(hostname, NULL, &hint, &aip) != 0) {
return 0;
}
sinp = (struct sockaddr_in *) aip->ai_addr;
ip = *(unsigned *) &sinp->sin_addr;
(Раньше я получал список из 3-х аддринфо с тремя SOCK_STREAM, SOCK_DGRAM и SOCK_RAW, но подсказка предотвращает это)
Так что мой вопрос все еще стоит ...
Решение
Существует функция POSIX getaddrinfo ()
, которая возвращает связанный список адресов для данного имени хоста, поэтому вам просто нужно просмотреть этот список и найти адрес без обратной связи.
Смотрите man getaddrinfo
.
Другие советы
Не ответ, а соответствующий комментарий: имейте в виду, что как только вы начнете отправлять адресную информацию в содержимом пакетов, вы рискуете сделать ваше приложение неспособным работать через NAT: ing маршрутизаторы и / или через брандмауэры.
Эти технологии полагаются на информацию в заголовках IP-пакетов для отслеживания трафика, и если приложения обмениваются адресной информацией внутри пакетов, где они остаются невидимыми для этой проверки, они могут сломаться.
Конечно, это может быть совершенно неуместно для вашего приложения, но я подумал, что на это стоит обратить внимание в этом контексте.
Исходный адрес будет включен в отправленный пакет ... нет необходимости дублировать эту информацию. Он получен при приеме сообщения от удаленного хоста (см. Руководство beej по работе с сетями, в частности, часть на accept () )
Я только что столкнулся с ситуацией, когда информация содержится только в / etc / hosts, и когда я использовал getaddrinfo для получения списка IP-адресов, он каждый раз возвращал 127.0.0.1. Как оказалось, имя хоста было псевдонимом localhost ... что-то, что часто легко пропустить. Вот что случилось:
Файл / etc / hosts:
127.0.0.1 localhost.localdomain localhost foo
:: 1 localhost6.localdomain6 localhost6
172.16.1.248 foo
172.16.1.249 bie
172.16.1.250 bletch
Итак, теперь, когда вы вызываете getaddrinfo с host = " foo " ;, он возвращает 127.0.0.1 3 раза. Ошибка здесь заключается в том, что foo появляется в строке с «127.0.0.1». и "172.16.1.248". Однажды я удалил foo из строки с «127.0.0.1» все работало нормально.
Надеюсь, это кому-нибудь поможет.
Посмотрите на это: Программное обнаружение общедоступных IP-адресов
Обратите внимание, что в некоторых случаях у компьютера может быть несколько IP-адресов без обратной связи, и в этом случае ответы на этот вопрос говорят о том, как получить тот, который доступен в Интернете.
Даже если компьютер имеет только один физический сетевой интерфейс (предположение, которое может или не может быть, даже у нетбуков есть два - Ethernet и WLAN), VPN может добавить еще больше IP-адресов. В любом случае, хост с другой стороны должен быть в состоянии определить IP-адрес вашего хоста, с которым он связался.
Используйте getaddrinfo ()
Ты почти у цели. Я не уверен, как вы получаете my_ip
из hp
.
gethostbyname ()
возвращает указатель на структуру hostent
, которая имеет поле h_addr_list
.
Поле h_addr_list
представляет собой список с нулевым символом в конце всех IP-адресов, привязанных к этому хосту.
Я думаю, вы получаете адрес обратной связи, потому что это первая запись в h_addr_list
.
РЕДАКТИРОВАТЬ: должно работать примерно так:
gethostname(myhostname, 32);
hp = gethostbyname(myhostname);
unsigned my_ip = *(unsigned*)(hp->h_addr);
for (int i = 0; hp->h_addr_list[i] != 0; ++i) {
if (hp->h_addr_list[i] != INADDR_LOOPBACK) {
// hp->addr_list[i] is a non-loopback address
}
}
// no address found
Если / etc / hosts все еще там и все тот же, поиск всех записей в h_addr_list не поможет.
Ваш новый код использует IPv4 (в поле hint.ai_family), что является ужасной идеей.
Кроме того, вы близки, вам просто нужно просмотреть результаты getaddrinfo. Ваш код только получает первый IP-адрес, но есть поле aip-> ai_next, за которым нужно следовать ...
struct addrinfo {
...
struct addrinfo *ai_next; /* next structure in linked list */
};