У меня есть такой код:
if (!socket_connect($this->sock, $this->host, $this->port)) {
где $this->host
это имя хоста, которое разрешается в 2 IP-адреса.
Если я strace
этот код я вижу что-то вроде
1456868407.567615 socket(PF_INET, SOCK_STREAM, IPPROTO_IP) = 3
1456868407.567805 fcntl(3, F_GETFL) = 0x2 (flags O_RDWR)
1456868407.568081 fcntl(3, F_SETFL, O_RDWR|O_NONBLOCK) = 0
1456868407.568264 connect(3, {sa_family=AF_INET, sin_port=htons(5672), sin_addr=inet_addr("192.168.1.111")}, 16) = -1 EINPROGRESS (Operation now in progress)
1456868407.568763 poll([{fd=3, events=POLLIN|POLLOUT|POLLERR|POLLHUP}], 1, 3000) = 1 ([{fd=3, revents=POLLIN|POLLOUT|POLLERR|POLLHUP}])
1456868408.724034 getsockopt(3, SOL_SOCKET, SO_ERROR, [111], [4]) = 0
1456868408.724699 fcntl(3, F_SETFL, O_RDWR) = 0
1456868408.725414 close(3) = 0
1456868408.725901 socket(PF_INET, SOCK_STREAM, IPPROTO_IP) = 3
1456868408.726408 fcntl(3, F_GETFL) = 0x2 (flags O_RDWR)
1456868408.727032 fcntl(3, F_SETFL, O_RDWR|O_NONBLOCK) = 0
1456868408.727727 connect(3, {sa_family=AF_INET, sin_port=htons(5672), sin_addr=inet_addr("192.168.1.112")}, 16) = -1 EINPROGRESS (Operation now in progress)
1456868408.728484 poll([{fd=3, events=POLLIN|POLLOUT|POLLERR|POLLHUP}], 1, 1842) = 1 ([{fd=3, revents=POLLOUT}])
1456868408.729281 getsockopt(3, SOL_SOCKET, SO_ERROR, [0], [4]) = 0
1456868408.729590 fcntl(3, F_SETFL, O_RDWR) = 0
Итак, после первого обращения 192.168.1.111
возвращенный ECONNREFUSED
(111
) для соединения — php пытается следующий адрес.
Я проверил соответствующий исходный код php для версии, с которой я имею дело: https://github.com/php/php-src/blob/php-5.5.9/ext/sockets/sockets.c#L1376 и не нашли нигде места, где это петли
Разрешает доменное имя через if (! php_set_inet_addr(&sin, addr, php_sock TSRMLS_CC)) {
Который, в свою очередь, устанавливает только первый адрес из возвращенного hostent
состав memcpy(&(sin->sin_addr.s_addr), host_entry->h_addr_list[0], host_entry->h_length);
в https://github.com/php/php-src/blob/php-5.5.9/ext%2Fsockets%2Fsockaddr_conv.c#L106
Поэтому мой вопрос: где реализован соответствующий код?
Или это делает не php, а glibc? Если так — где это в источниках glibc? (Я тоже пытался найти там, но не удалось)
Нашел это: https://github.com/php/php-src/blob/php-5.5.9/main/network.c#L797
for (sal = psal; !fatal && *sal != NULL; sal++) {
sa = *sal;
/* create a socket for this address */
sock = socket(sa->sa_family, socktype, 0);
if (sock == SOCK_ERR) {
continue;
}
// <skipped>
n = php_network_connect_socket(sock, sa, socklen, asynchronous,
timeout ? &working_timeout : NULL,
error_string, error_code);
if (n != -1) { // <-- here is the check
goto connected;
}
// skipped
}
здесь он перебирает адреса, пока не сможет подключиться к одному.
Других решений пока нет …