以下に、クライアントとサーバーの間で WebSocket 接続を確立するときのハンドシェイク部分を示す図を描きました。これは、ノードによって提供されるネット モジュールが開発者によってすでにカプセル化されているため、この部分はノード内で非常に簡単に完了できます。接続の確立に対処する必要はなく、データの相互作用のみを考慮する必要があります。ただし、PHPはそうではなく、ソケットの接続、確立、バインディング、監視などを自分で操作する必要があるため、取り出して話す必要があります。
①と②は実際にはHTTPリクエストとレスポンスですが、処理中に取得するのは解析されていない文字列です。例:
通常、このようなリクエストがサーバーに到達すると、コード ライブラリを通じてこの情報を直接取得できます。
1. phpでWebSocketを処理する
WebSocket 接続はクライアントによってアクティブに開始されるため、すべてはクライアントから開始する必要があります。最初のステップは、クライアントから送信された Sec-WebSocket-Key 文字列を解析することです。
クライアントリクエストフォーマット
まず、php はソケット接続を確立し、ポート情報をリッスンします。
1.ソケット接続の確立
ソケットの確立については、大学でコンピュータ ネットワークを学んだことのある多くの人が知っていると思います。次は、接続を確立するプロセスの図です。
function __construct($address, $port){
// 建立一个 socket 套接字
$this->master = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)
or die("socket_create() failed");
socket_set_option($this->master, SOL_SOCKET, SO_REUSEADDR, 1)
or die("socket_option() failed");
socket_bind($this->master, $address, $port)
or die("socket_bind() failed");
socket_listen($this->master, 2)
or die("socket_listen() failed");
$this->sockets[] = $this->master;
// debug
echo("Master socket : ".$this->master."\n");
while(true) {
//自动选择来消息的 socket 如果是握手 自动选择主机
$write = NULL;
$except = NULL;
socket_select($this->sockets, $write, $except, NULL);
foreach ($this->sockets as $socket) {
//主机のクライアントに接続します
if ($socket == $this->master){
$client =ソケット_accept($this->master);
if ($client < 0) {
// debug
echo "socket_accept() が失敗しました";
continue;
} else {
//connect($client);
array_push($this->sockets, $ client);
echo "connect clientn";
}
} else {
$bytes = @socket_recv($s ocket,$buffer,2048,0);
if($bytes == 0) return;
if (!$ this->handshake) {
// 如果没有手,先手握回应
// doHandShake($socket, $buffer);
echo "shakeHandsn";
} else {
// 如果已经握り手、直接受信データ、并处理
$buffer = decode($buffer);
//process($socket, $buffer);
echo "ファイルを送信";
}
}
}
}
}
}
上のこのセグメントコードは私が承認済みであり、大した問題はありませんが、必要に応じて cmd 実行コマンドで php /path/to/demo.php にアクセスできます。当然、上だけが 1 つの種類であり、必要な場合は
新しい例も入手します。客户端代网可以稍微简单点:
クライアントが接続すると、サーバー コードを実行します。
2. Sec-WebSocket-Key 情報を抽出します
コードをコピーします コードは次のとおりです:
ここで必ず注意してください。すべてのリクエストと対応する形式には最後に空白行があります。これは、テストを開始したときに失われ、長い間苦労しました。
5. データフレームの処理
コードをコピー
コードは次のとおりです:
}
for ($index = 0; $index < strlen($data) ); $index++) {
$decoded .= $data[$index] ^ $masks[$index % 4] ;
}
return $decoded;
}
ここで関係するエンコーディングの問題については、前の記事で述べたので、ここでは詳しく説明しません。PHP には文字処理に関する関数が多すぎて、詳しくは覚えていません。ここではデコードプログラムを作成し、クライアントに直接データを送信します。これはチャット ルームのモデルとみなすことができます。
クライアントコード:
コードは次のとおりです:
2. 問題に注意してください
ハンドシェイク中のクライアントのリクエストには、バージョン識別子である Sec-WebSocket-Version: 13 が含まれています。これは、現在のすべてのブラウザーがこのバージョンを使用します。以前のバージョンは、データ暗号化部分でさらに面倒でした:
コードをコピーします
コードは次のとおりです:
コードは次のとおりです:
if($key1_spc==0|$key2_spc==0){ $this->log("無効なキー");return; }
//いくつかの計算
$key1_sec = Pack("N",$key1_num / $key1_spc);
$key2_sec = Pack("N",$key2_num / $key2_spc);
}
コードは次のとおりです:
2. データフレーム解析コード
この記事では、decodeFrame などのデータ フレーム解析コードは提供しません。データ フレームの形式は、純粋に物理的な作業です。