Heim  >  Artikel  >  Backend-Entwicklung  >  Detaillierte Erklärung der PHP-Implementierung von Ping (Raw Socket) über das ICMP-Protokoll

Detaillierte Erklärung der PHP-Implementierung von Ping (Raw Socket) über das ICMP-Protokoll

藏色散人
藏色散人nach vorne
2021-03-31 17:24:332928Durchsuche

Empfohlenes Lernen: „PHP-Video-Tutorial

PHP implementiert Ping (Raw Socket) über das ICMP-Protokoll

Ich wollte kürzlich eine Funktion implementieren, um zu erkennen, ob der Zielhost online ist, und habe dies überprüft Mit Baidu wird meist eine zu einem bestimmten Port geöffnete Verbindung verwendet, um festzustellen, ob der Zielhost online ist. Beispielsweise Port 3389 (RDP) von Windows-Systemen und Port 22 (SSH) von *nix-Systemen.

Wenn der Zielhost diese Ports jedoch nicht öffnet, führt dies zu Fehlern bei der Beurteilung. Nur weil ein bestimmter Port nicht geöffnet ist, heißt das nicht, dass der Zielhost offline ist.

Da die meisten Geräte auf Ping reagieren, habe ich mir überlegt, diese Funktion mit Ping zu implementieren. Als ich Baidu erneut befragte, stellte ich fest, dass die meisten Tutorials die Funktion exec() verwenden, um den System-Ping-Befehl aufzurufen, was offensichtlich sehr unsicher ist.

Also habe ich mich schließlich entschieden, den von PHP bereitgestellten Raw-Socket zu verwenden und das ICMP-Paket selbst zu erstellen, um Ping zu implementieren.

Um ein ICMP-Paket zu erstellen, müssen wir zunächst die Struktur des ICMP-Pakets verstehen.

Detaillierte Erklärung der PHP-Implementierung von Ping (Raw Socket) über das ICMP-Protokoll

Wie Sie sehen können, besteht ein Standard-ICMP-Paket aus 8-Bit-Typ, 8-Bit-Code, 16-Bit-Prüfsumme, 16-Bit-ID, 16-Bit-Sequenznummer und Daten. Als nächstes erstellen wir ein solches Datenpaket über PHP.

$package = chr(8).chr(0);//模式 8 0
$package .= chr(0).chr(0);//置零校验和
$package .= "R"."C";//ID 这里是我随便填的
$package .= chr(0).chr(1);//序列号 一样 随便填的
for($i=strlen($package);$i<64;$i++){//填充满64位
    $package .= chr(0);//数据
}

Als nächstes berechnen Sie die Prüfsumme.

$tmp = unpack("n*",$package);//把数据16位一组放进数组里
$sum = array_sum($tmp);//求和
$sum = ($sum >> 16) + ($sum & 0xFFFF);//结果右移十六位 加上结果与0xFFFF做AND运算
$sum = $sum + ($sum >> 16);//结果加上结果右移十六位
$sum = ~ $sum;//做NOT运算
$checksum = pack("n*", $sum);//打包成2字节

Fügen Sie die Prüfsumme in das Paket ein.

$package[2] = $checksum[0];
$package[3] = $checksum[1];//填充校验和

Auf diese Weise wird ein Standard-ICMP-Datenpaket erstellt und kann direkt an den Zielhost gesendet werden. Bereit zu gehen ~

$host = "192.168.1.1";//设置目标主机
$socket=socket_create(AF_INET, SOCK_RAW, getprotobyname(&#39;icmp&#39;));//创建原始套接字
$start = microtime();//记录开始时间
socket_sendto($socket, $package, strlen($package), 0, $host, 0);//发送数据包
$read = array($socket);//初始化socket
$select = socket_select($read, $write, $except, 5);
if ($select === FALSE){
    $icmpError = "socket_select()方法发生错误,原因:".socket_strerror(socket_last_error());
    socket_close($socket);
}else if($select === 0){
    $icmpError = "请求超时";
    socket_close($socket);
}
if($icmpError !== NULL){
    echo $icmpError;
    exit();
}
socket_recvfrom($socket, $recv, 65535, 0, $host, $port);//接受回传数据
/*回传数据处理*/
$end = microtime();//记录结束时间
$recv = unpack("C*", $recv);
$length = count($recv) - 20;//包长度 减去20字节IP报头
$ttl = $recv[9];//ttl
$seq = $recv[28];//序列号
$duration = round(($end - $start) * 1000,3);//计算耗费的时间
echo "{$length} bytes from {$host}: icmp_seq={$seq}  ttl={$ttl} time={$duration}ms".PHP_EOL;//输出结果

Zum Ausführen tippen, eine Ping-Anfrage ist abgeschlossen. Wenn nichts anderes passiert, sollte das Ergebnis wie unten dargestellt aussehen.

64 bytes from 192.168.1.1: icmp_seq=1  ttl=128 time=0.589ms

Schließlich habe ich diesen Code in eine Funktion gepackt. Fügen Sie es Ihrem Code hinzu und verwenden Sie ping(string $host, int $retry), wenn Sie es aufrufen müssen.

<?php
 function ping($host, $retry = 1){
    $g_icmp_error = NULL;
    $write = NULL;
    $except = NULL;//初始化所需变量
    $package = chr(8).chr(0);//模式 8 0
    $package .= chr(0).chr(0);//置零校验和
    $package .= "R"."C";//ID
    $package .= chr(0).chr(1);//序列号
    for($i=strlen($package);$i<64;$i++){
        $package .= chr(0);
    }
    $tmp = unpack("n*",$package);//把数据16位一组放进数组里
    $sum = array_sum($tmp);//求和
    $sum = ($sum >> 16) + ($sum & 0xFFFF);//结果右移十六位 加上结果与0xFFFF做AND运算
    $sum = $sum + ($sum >> 16);//结果加上结果右移十六位
    $sum = ~ $sum;//做NOT运算
    $checksum = pack("n*", $sum);//打包成2字节
    $package[2] = $checksum[0];
    $package[3] = $checksum[1];//填充校验和
    $socket=socket_create(AF_INET, SOCK_RAW, getprotobyname(&#39;icmp&#39;));//创建原始套接字
    $start = microtime();//记录开始时间
    socket_sendto($socket, $package, strlen($package), 0, $host, 0);//发送数据包
    $read = array($socket);//初始化socket
    $select = socket_select($read, $write, $except, 5);
    if ($select === FALSE){
        $icmpError = "socket_select()方法发生错误,原因:".socket_strerror(socket_last_error());
        socket_close($socket);
    }else if($select === 0){
        $icmpError = "请求超时";
        socket_close($socket);
    }
    if($icmpError !== NULL){
        echo $icmpError;
        exit();
    }
    socket_recvfrom($socket, $recv, 65535, 0, $host, $port);//接受回传数据
    /*回传数据处理*/
    $end = microtime();//记录结束时间
    $recv = unpack("C*", $recv);
    $length = count($recv) - 20;//包长度 减去20字节IP报头
    $ttl = $recv[9];//ttl
    $seq = $recv[28];//序列号
    $duration = round(($end - $start) * 1000,3);//计算耗费的时间
    echo "{$length} bytes from {$host}: icmp_seq={$seq}  ttl={$ttl} time={$duration}ms".PHP_EOL;//输出结果
    
    socket_close($socket);//关闭socket
}
?>

Wenn der Artikel Fehler oder fehlende Details enthält, weisen Sie ihn bitte darauf hin und diskutieren Sie ihn im Kommentarbereich.

Das obige ist der detaillierte Inhalt vonDetaillierte Erklärung der PHP-Implementierung von Ping (Raw Socket) über das ICMP-Protokoll. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:joyrunc. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen