Home >Backend Development >PHP Tutorial >PHP practice of building a chat room based on websocket

PHP practice of building a chat room based on websocket

墨辰丷
墨辰丷Original
2018-05-31 10:43:217253browse

This article mainly explains the practice of building a simple chat room with php websocket. Please refer to the content of php and websocket in this article for those who need it.

The example in this article describes the practice of building a simple chat room in PHP based on websocket. Share it with everyone for your reference. The details are as follows:
1. Preface

There is a simple chat room in the company's game. After understanding it, I realized that it was made by node websocket. I thought about making a simple chat room with PHP. chat room. So I collected various information, read documents, looked for examples, and wrote a simple chat room myself.

http connections are divided into short connections and long connections. Short connections can generally be implemented using ajax, and long connections are websockets. Short connections are relatively simple to implement, but consume too many resources. Websocket is efficient but has some issues with compatibility. Websocket is a resource of html5

2. Front-end

The front-end implementation of websocket is very simple and straightforward

//连接websocket

var ws = new WebSocket("ws://127.0.0.1:8000");

//成功连接websoc的时候

ws.onopen = function(){}

//成功获取服务端输出的消息

ws.onmessage = function(e){}

//连接错误的时候
ws.onerror = function(){}

//向服务端发送数据

ws.send();

3. Backend

The difficulty of websocket is mainly in the background

3.1websocket connection process
websocket communication diagram This is a simple client Communication diagram between the client and the server. What PHP mainly does is to accept the encryption key and return it to complete the creation and handshake of the socket.

The following picture is a A detailed flow chart of the server processing websocket

3.2 Code practice

The process done by the server is roughly:

  1. Hang a socket process waiting for connection

  2. Traverse the socket array after there is a socket connection

  3. If there is no handshake, the handshake operation is performed. If there is a handshake, the data is parsed and written into the buffer for output

The following is a sample code (I wrote a class Therefore, the code is segmented according to functions). The github address and some pitfalls encountered at the end of the article are given
1. First, create the socket

//建立套接字
    public function createSocket($address,$port)
    {
      //创建一个套接字
      $socket= socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
      //设置套接字选项
      socket_set_option($socket, SOL_SOCKET, SO_REUSEADDR, 1);
      //绑定IP地址和端口
      socket_bind($socket,$address,$port);
      //监听套接字
      socket_listen($socket);
      return $socket;
    }

2. Put the socket into the array

public function __construct($address,$port)
    {
      //建立套接字
      $this->soc=$this->createSocket($address,$port);
      $this->socs=array($this->soc);

    }

3. Suspend the process to traverse the socket array, main operations It’s all done here

public function run(){
      //挂起进程
      while(true){
        $arr=$this->socs;
        $write=$except=NULL;
        //接收套接字数字 监听他们的状态
        socket_select($arr,$write,$except, NULL);
        //遍历套接字数组
        foreach($arr as $k=>$v){
          //如果是新建立的套接字返回一个有效的 套接字资源
          if($this->soc == $v){
            $client=socket_accept($this->soc);
            if($client <0){
              echo "socket_accept() failed";
            }else{
              // array_push($this->socs,$client);
              // unset($this[]);
              //将有效的套接字资源放到套接字数组
              $this->socs[]=$client;
            }
          }else{
            //从已连接的socket接收数据 返回的是从socket中接收的字节数
            $byte=socket_recv($v, $buff,20480, 0);
            //如果接收的字节是0
            if($byte<7)
              continue;
            //判断有没有握手没有握手则进行握手,如果握手了 则进行处理
            if(!$this->hand[(int)$client]){
              //进行握手操作
              $this->hands($client,$buff,$v);
            }else{
              //处理数据操作
              $mess=$this->decodeData($buff);
                //发送数据
              $this->send($mess,$v);
            }
          }
        }
      }
    }

4. The handshake process is to receive the websocket content and obtain the key from Sec-WebSocket-Key: and pass The client will verify the encryption algorithm written into the buffer (automatic verification does not require our processing)

public function hands($client,$buff,$v)
    {
      //提取websocket传的key并进行加密 (这是固定的握手机制获取Sec-WebSocket-Key:里面的key)
      $buf = substr($buff,strpos($buff,&#39;Sec-WebSocket-Key:&#39;)+18);
      //去除换行空格字符
      $key = trim(substr($buf,0,strpos($buf,"\r\n")));
       //固定的加密算法
      $new_key = base64_encode(sha1($key."258EAFA5-E914-47DA-95CA-C5AB0DC85B11",true));
      $new_message = "HTTP/1.1 101 Switching Protocols\r\n";
      $new_message .= "Upgrade: websocket\r\n";
      $new_message .= "Sec-WebSocket-Version: 13\r\n";
      $new_message .= "Connection: Upgrade\r\n";
      $new_message .= "Sec-WebSocket-Accept: " . $new_key . "\r\n\r\n";
      //将套接字写入缓冲区
      socket_write($v,$new_message,strlen($new_message));
      // socket_write(socket,$upgrade.chr(0), strlen($upgrade.chr(0)));
      //标记此套接字握手成功
      $this->hand[(int)$client]=true;
    }

5. Parse the client's data (I am here There is no encryption, you can encrypt it yourself if necessary)

//解析数据
    public function decodeData($buff)
    {
      //$buff 解析数据帧
      $mask = array(); 
      $data = &#39;&#39;; 
      $msg = unpack(&#39;H*&#39;,$buff); //用unpack函数从二进制将数据解码
      $head = substr($msg[1],0,2); 
      if (hexdec($head{1}) === 8) { 
        $data = false; 
      }else if (hexdec($head{1}) === 1){ 
        $mask[] = hexdec(substr($msg[1],4,2)); 
        $mask[] = hexdec(substr($msg[1],6,2)); 
        $mask[] = hexdec(substr($msg[1],8,2)); 
        $mask[] = hexdec(substr($msg[1],10,2)); 
          //遇到的问题 刚连接的时候就发送数据 显示 state connecting
        $s = 12; 
        $e = strlen($msg[1])-2; 
        $n = 0; 
        for ($i=$s; $i<= $e; $i+= 2) { 
          $data .= chr($mask[$n%4]^hexdec(substr($msg[1],$i,2))); 
          $n++; 
        }
        //发送数据到客户端
          //如果长度大于125 将数据分块
          $block=str_split($data,125);
          $mess=array(
            &#39;mess&#39;=>$block[0],
            );
        return $mess;          
      }

6. Write the socket to the buffer

//发送数据
    public function send($mess,$v)
    {
      //遍历套接字数组 成功握手的 进行数据群发
      foreach ($this->socs as $keys => $values) {
        //用系统分配的套接字资源id作为用户昵称
          $mess[&#39;name&#39;]="Tourist&#39;s socket:{$v}";
          $str=json_encode($mess);
          $writes ="\x81".chr(strlen($str)).$str;
          // ob_flush();
          // flush();
          // sleep(3);
          if($this->hand[(int)$values])
            socket_write($values,$writes,strlen($writes));
        }
    }

7. Running method

github address git@github.com:rsaLive/websocket.git

①It is best to control Run server.php

Go to the server.php script directory (you can first php -v to see if php is configured. If there is no Linux configuration, configure the path under bash windows)

php -f server.php

If there is an error, it will prompt

##②Access the html file through the server

#8. For pitfalls that have been stepped on, open debugging to facilitate viewing errors

①server.php can print output in the suspended process. If a problem occurs, you can add printing to the code for debugging.

You can mark each judgment and check on the console which section the code is running in.

But you need to re-run the script every time you modify the code. php server.php

② If this error occurs, it may be

1. Send data when initializing the socket with the server (in the first Content cannot be sent during the first verification handshake with the server)

2. This will also happen if the client has verified but the client has not sent it or the message sent is empty

So you need to check Connected socket data

③The browser may not support it or the server may not have opened the socket. It is best to verify before starting

if (window.WebSocket){
  console.log("This browser supports WebSocket!");
} else {
  console.log("This browser does not support WebSocket.");
}

The above is the entire content of this article, I hope it will be helpful to everyone's study.


Related recommendations:

Detailed explanation of PHP random red envelope algorithm

Do it with PHP Exit the search for nearby people function

PHP long connection use case analysis

The above is the detailed content of PHP practice of building a chat room based on websocket. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn