2013-09-30 2 views
1

У меня есть скрипт PHP (который я нашел в Интернете), который бесконечно слушает порт, и если есть подключение к этому порту, оно установит TCP connection. Однако, когда я запускал этот скрипт и было много соединений (около 500), число подключений CLOSE_WAIT увеличивается. Удаленное устройство, подключенное в этом состоянии, не может подключиться снова, потому что CLOSE_WAIT не завершалось.Сценарий сокета PHP вызывает многократное соединение сокета CLOSE_WAIT

// port info 
$host = "0.0.0.0"; 
$port = 10260; 
$pos = 1; 

// don't timeout! 
set_time_limit(0); 
record("START"); 

$sock = socket_create(AF_INET, SOCK_STREAM, 0); 
$timeout = array('sec'=>3000,'usec'=>0); 
$try = socket_set_option($sock,SOL_SOCKET,SO_RCVTIMEO,$timeout); 

// Bind the socket to the address/port 
if(!socket_bind($sock, $host, $port)) 
{ 
    echo socket_last_error() ; 
    die('Could not bind to address'); 
} 
record("SOCKET BIND OK"); 

// start listening for connections 
$result = socket_listen($sock, 1024) or die("Could not set up socket listener\n"); 
record("SOCKET LISTEN OK"); 
$clients = array($sock); 


// infinite while loop 
while(1) 
{ 

    // Setup clients listen socket for reading 
    $read = $clients; 

    $e = NULL; 

    if (socket_select($read, $write = NULL, $except = NULL, 0,0) < 1) 
    { 
     continue; 
    } 

    /* if a new read ready is being made add it to the client array */ 

    if (in_array($sock, $read)) { 
     record("NEW CONNECTION"); 
     $clients[$pos] = $newsock = socket_accept($sock); 
     $curpos = $pos; 
     $pos++; 
     socket_getpeername($newsock, $ip,$port); 
     record("Incoming IP: {$ip} PORT: {$port}"); 
     // remove the listening socket from the clients-with-data array 
     $key = array_search($sock, $read); 
     unset($read[$key]); 

    } // end if in_array 

    // loop through all the clients that have data to read from 
    foreach ($read as $read_key => $read_sock) { 
     // read until newline or 1024 bytes 
     // socket_read while show errors when the client is disconnected, so silence the error messages 
     $key = $read_key; 
     $fulldata = $data = @socket_read($read_sock, 1024); 

     // check if the client is disconnected 
     if ($data === false) { 
      // remove client for $clients array 
      $key = array_search($read_sock, $clients); 
      socket_close($read_sock); 
      unset($clients[$key]); 

      record("NO DATA"); 

      // continue to the next client to read from, if any 
      continue; 
     } 


     // .. do something with $data ... 

    } 


} 

socket_close($sock);  
record("END"); 
die("DONE"); 

Я пытался использовать socket_close() в коде, но безрезультатно. CLOSE_WAIT, кажется, занимает больше времени, чтобы перейти к следующему состоянию.

+0

У меня была аналогичная проблема. В моем случае проблема заключалась в том, что один клиент, который внезапно отключился и когда его буфер был заполнен, больше ничего не мог подключиться к порту. Когда-либо понимал проблему, но я исправил с помощью ручной системы ping с меньшей терпимостью (просто информация, которая может вам помочь) –

+0

@chumkiu благодарит за информацию. Когда вы говорите, что буфер клиента заполнен, вы имеете в виду приложение или удаленное устройство, которое подключается к порту? Если это приложение, есть ли способ определить, заполнен ли буфер или нет? Благодаря! –

+0

Я подразумеваю 'Recv-Q' для призрачного клиента в' netstat' на сервере. Я ничего не нашел для чтения этого в php (но мой опыт - 6 лет назад) –

ответ

3

Я нашел две проблемы в вашем коде:

Во-первых, четвёртую параметра (tv_sec) из socket_select должен быть NULL вместо 0. Как описано в Manual, 0 причин высокой нагрузки процессора в бесполезную, бесконечный цикл , в то время как NULL будет блокироваться до тех пор, пока ничего не произойдет:

tv_sec может быть нулевой, вызывая немедленное возвращение socket_select(). Это полезно для опроса. Если tv_sec равно NULL (без тайм-аута), socket_select() может блокироваться бесконечно.

Во-вторых, и я думаю, что это причина для CLOSE_WAIT - это ваш тест, если $ data является ложным. Это работает только на изящном закрытом соединении, но, как говорится в одном из комментариев here, вы также проводите тест на пустую строку. Так изменить одну строку:

if (socket_select($read, $write = NULL, $except = NULL, NULL, 0) < 1) 

, а другой:

// check if the client is disconnected 
if ($data === false || $data === '') { 

и вы должны быть хорошо

+0

спасибо за указание ошибок. Я исправил их, и это помогло много. Состояние 'CLOSE_WAIT' переместилось на' LAST_ACK', когда соединение составляет около 100-200. Однако, когда он достигает около 500, я все еще получаю несколько соединений CLOSE_WAIT (намного меньше, чем раньше) с прерывистым состоянием 'LAST_ACK'. Это потому, что для завершения 'socket_close' требуется время? –

+0

В зависимости от того, что делает ваш скрипт со всеми клиентами, это может занять некоторое время, пока socket_close не будет вызван да, в это время он все еще находится в состоянии CLOSE_WAIT. Насколько я помню, LAST_ACK означает, что клиент ждет отправки ACK в пакет FIN. Если один из этих пакетов есть в брандмауэре или что-то в этом роде, вам нужно подождать, пока ядро ​​не очистит соединение таймаутом. – a4c8b

Смежные вопросы