WorkerMan 入门学习之(四)GatewayWorker框架与ThinkPHP5.1框架结合案例
GatewayWorker是基于Workerman开发的一个可分布式部署的TCP长连接框架,专门用于快速开发TCP长连接应用,例如app推送服务端、即时IM服务端、游戏服务端、物联网、智能家居等等
文档地址:http://www.workerman.net/gatewaydoc/
一、测试官方DEMO(Windows 版本)
1、下载demo
2、解压到任意位置,我这里为:D:\phpStudy\PHPTutorial\WWW\GatewayWorker
3、进入GatewayWorker目录
4、双击start_for_win.bat启动。(如果出现错误请参考这里设置php环境变量),效果如下
5、命令行窗口运行 telnet 127.0.0.1 8282
,输入任意字符即可聊天(非本机测试请将127.0.0.1替换成实际ip)。
PS:以上表示TCP连接测试成功
二、修改测试websocket
1、需要修改 start_gateway.php 指定websocket协议,像这样
1 | $gateway = new Gateway(websocket: //0.0.0.0:7272); |
2、重新启动 start_for_win.bat
3、测试js
小结:只需要改动一个文件( start_gateway.php
)的协议和端口即可,别的不需用改动。
三、与ThinkPHP5.1框架结合
(一)服务端主动推送消息到客户端
原则:
1、TP5.1框架项目与GatewayWorker独立部署互不干扰
2、所有的业务逻辑都由网站(websocket连接的)页面以post/get请求到TP5.1框架的控制器中完成
3、GatewayWorker不接受客户端发来的数据,即GatewayWorker不处理任何业务逻辑,GatewayWorker仅仅当做一个单向的推送通道
4、仅当TP5.1框架需要向浏览器主动推送数据时才在TP5.1框架中调用Gateway的API(GatewayClient)完成推送
具体实现步骤
1、网站页面建立与GatewayWorker的websocket连接
1 | ws = new WebSocket( "ws://127.0.0.1:7272" ); |
2、GatewayWorker发现有页面发起连接时,将对应连接的client_id发给网站页面
Event.php 内容
1 2 3 4 5 6 7 8 9 | public static function onConnect( $client_id ) { $resData = [ 'type' => 'init' , 'client_id' => $client_id , 'msg' => 'connect is success' // 初始化房间信息 ]; Gateway::sendToClient( $client_id , json_encode( $resData )); } |
index.html 内容
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | <!DOCTYPE html> <html lang= "en" > <head> <meta charset= "UTF-8" > <title>GatewayWorker的websocket连接</title> </head> <body> <h1>GatewayWorker的websocket连接</h1> <script src= "https://cdn.bootcss.com/jquery/3.3.1/jquery.min.js" ></script> <script type= "text/javascript" > ws = new WebSocket( "ws://127.0.0.1:7272" ); // 服务端主动推送消息时会触发这里的onmessage ws.onmessage = function (e){ // json数据转换成js对象 var data = JSON.parse(e.data); console.log(data); var type = data.type || '' ; switch (type){ // Events.php中返回的init类型的消息,将client_id发给后台进行uid绑定 case 'init' : // 利用jquery发起ajax请求,将client_id发给后端进行uid绑定 $.post( "{:url('index/chat_room/bind')}" , {client_id: data.client_id}, function (data) { console.log(data); }, 'json' ); break ; case 'say' : console.log( 'TP5 msg' +e.data); break ; // 当mvc框架调用GatewayClient发消息时直接alert出来 default : alert(e.data); } }; </script> </body> </html> |
3、网站页面收到client_id后触发一个ajax请求(index/chat_room/bind)将client_id发到TP5.0后端,bind方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | /* * 用户登录后初始化以及绑定client_id */ public function bind() { // 设置GatewayWorker服务的Register服务ip和端口,请根据实际情况改成实际值 Gateway:: $registerAddress = '127.0.0.1:1238' ; $uid = $this ->userId; $group_id = $this ->groupId; $client_id = request()->param( 'client_id' ); // client_id与uid绑定 Gateway::bindUid( $client_id , $uid ); // 加入某个群组(可调用多次加入多个群组) Gateway::joinGroup( $client_id , $group_id ); } |
4、后端收到client_id后利用GatewayClient调用Gateway::bindUid($client_id, $uid)
将client_id与当前uid(用户id或者客户端唯一标识)绑定。如果有群组、群发功能,也可以利用Gateway::joinGroup($client_id, $group_id)
将client_id加入到对应分组
连接成功后返回值
PS:以上返回值为 GatewayWorker服务 连接成功后返回的json数据
5、页面发起的所有请求都直接post/get到mvc框架统一处理,包括发送消息
通过sendMessage发送消息(服务端主动推送消息到客户端)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | // mvc后端发消息 利用GatewayClient发送 Events.php public function sendMessage() { // stream_socket_client(): unable to connect to tcp://127.0.0.1:1236 $uid = $this ->userId; $group = $this ->groupId; $message = json_encode([ 'type' => 'say' , 'msg' => 'Hello ThinkPHP5' ]); // 设置GatewayWorker服务的Register服务ip和端口,请根据实际情况改成实际值 Gateway:: $registerAddress = '127.0.0.1:1238' ; // 向任意uid的网站页面发送数据 Gateway::sendToUid( $uid , $message ); // 向任意群组的网站页面发送数据,如果开启,则会向页面发送两条一样的消息 //Gateway::sendToGroup($group, $message); } |
6、mvc框架处理业务过程中需要向某个uid或者某个群组发送数据时,直接调用GatewayClient的接口Gateway::sendToUid Gateway::sendToGroup
等发送即可
通过浏览器访问sendMessage操作,测试结果
PS:以上的消息是TP5.0 通过 GatewayClient\Gateway 发送写消息,和GatewayWorker服务没有直接关系
以上为 服务端主动推送消息到客户端
注意区分:
1、服务端主动推送消息到客户端
2、客户端推送消息到客户端
(二)客户端推送消息到客户端
修改客户端到客户端的消息发送和接受,下面修改 GatewayWorker 的 Events.php(开发者只需要关注这个文件)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 | public static function onConnect( $client_id ) { $resData = [ 'type' => 'init' , 'client_id' => $client_id , 'msg' => 'connect is success' // 初始化房间信息 ]; Gateway::sendToClient( $client_id , json_encode( $resData )); } /** * 当客户端发来消息时触发 * @param int $client_id 连接id * @param mixed $message 具体消息 */ public static function onMessage( $client_id , $message ) { // 服务端console输出 //echo "msg : $message \r\n"; // 解析数据 $resData = json_decode( $message , true); $type = $resData [ 'type' ]; $roomId = $resData [ 'roomId' ]; $userId = $resData [ 'userId' ]; // 未登录,则传递一个随机 $userName = $resData [ 'userName' ]; // 未登录,则传递一个随机 $content = isset( $resData [ 'content' ]) ? $resData [ 'content' ] : 'default content' ; //将时间全部置为服务器时间 $serverTime = date ( 'Y-m-d H:i:s' , time()); switch ( $type ) { case 'join' : // 用户进入直播间 //将客户端加入到某一直播间 Gateway::joinGroup( $client_id , $roomId ); $resData = [ 'type' => 'join' , 'roomId' => $roomId , 'userName' => $userName , 'msg' => "enters the Room" , // 发送给客户端的消息,而不是聊天发送的内容 'joinTime' => $serverTime // 加入时间 ]; // 广播给直播间内所有人,谁?什么时候?加入了那个房间? Gateway::sendToGroup( $roomId , json_encode( $resData )); break ; case 'say' : // 用户发表评论 $resData = [ 'type' => 'say' , 'roomId' => $roomId , 'userName' => $userName , 'content' => $content , 'commentTime' => $serverTime // 发表评论时间 ]; // 广播给直播间内所有人 Gateway::sendToGroup( $roomId , json_encode( $resData )); break ; case 'pong' : break ; // 接收心跳 default : //Gateway::sendToAll($client_id,$json_encode($resData)); break ; } } |
index.html 聊天室页面
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 | <!DOCTYPE html> <html lang= "en" > <head> <meta charset= "UTF-8" > <title>GatewayWorker的websocket连接</title> </head> <body> <h1>GatewayWorker的websocket连接</h1> <div class = "row" > websocket send content:<input type= "text" style= "height: 50px; width: 100%;" name= "data" id= "data" > <p></p> <button id= "submit" onclick= "sub()" >send info</button> <p></p> <div id= "output" ></div> </div> <script src= "https://cdn.bootcss.com/jquery/3.3.1/jquery.min.js" ></script> <script src= "https://cdn.bootcss.com/reconnecting-websocket/1.0.0/reconnecting-websocket.min.js" ></script> <script language= "javascript" type= "text/javascript" > var wsUri = "ws://notes.env:7272/" ; var outputContent; var roomId = 'L06777' ; var userId = 4840043; var userName = 'Tinywan' + Math.random(); // 把当新链接的客户端加入到当前直播间,消息类型:{"type":"join","roomId":"1002","userId":"88","userName":"userName"} var joinContent = { "type" : "join" , "roomId" : roomId, "userId" : userId, "userName" : userName }; // 初始化页面操作 function init() { outputContent = document.getElementById( "output" ); initWebSocket(); } function initWebSocket() { websocket = new ReconnectingWebSocket(wsUri); websocket.onopen = function (evt) { onOpen(evt) }; websocket.onclose = function (evt) { onClose(evt) }; websocket.onmessage = function (evt) { onMessage(evt) }; websocket.onerror = function (evt) { onError(evt) }; } function onOpen(evt) { console.log( "CONNECTED" ); } // 接收数据 function onMessage(evt) { var data = eval( "(" + evt.data + ")" ); var type = data.type || '' ; switch (type) { case 'init' : // 把当新链接的客户端加入到当前直播间 console.log( '-------init--------' + data); websocket.send(JSON.stringify(joinContent)); writeToScreen( '<span style="color: blue;">RESPONSE: ' + evt.data + '</span>' ); break ; case 'join' : console.log( '-------join--------' + data); writeToScreen( '<span style="color: blue;"> ' + ' 新用户: ' + '</span>' + '<span style="color: red;"> ' + data.userName + '</span>' + '<span style="color: green;"> ' + data.joinTime + '</span>' + '<span style="color: black;"> ' + data.msg + '</span>' ); break ; case 'say' : console.log( 'say======' + data); writeToScreen( '<span style="color: blue;"> ' + ' Chat: ' + '</span>' + '<span style="color: red;"> ' + data.userName + '</span>' + '<span style="color: #D2691E;"> ' + data.commentTime + '</span>' + '<span style="color: black;"> ' + data.content + '</span>' ); break ; default : console.log(data); break ; } } function onError(evt) { console.log( '<span style="color: red;">ERROR:</span> ' + evt.data); } function onClose(evt) { console.log( "DISCONNECTED" ); } function writeToScreen(message) { var pre = document.createElement( "p" ); pre.style.wordWrap = "break-word" ; pre.innerHTML = message; outputContent.appendChild(pre); } function sub() { var text = document.getElementById( 'data' ).value; // {"type":"say",,"msg":"Welcome 111111111111Live Room"} var sayContent = { "type" : "say" , "roomId" : roomId, "userId" : userId, "userName" : userName, "content" : text }; websocket.send(JSON.stringify(sayContent)); } window.addEventListener( "load" , init, false ); </script> </body> </html> |
重启开启服务
测试结果
扩展:
可以把消息存储的Redis中,通过Redis统计直播间的PV
1 2 3 4 5 6 | $redis = new \Redis; $redis->connect( '127.0.0.1' ,6379); $key = "PV:ROOM:" .$roomId; $field = "ROOM_TOTAL_PV" ; // 进入房间的人数增长,自增 ,增加PV统计 $redis->hIncrBy($key,$field,1); |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
2017-06-09 NGINX 加载动态模块(NGINX 1.9.11开始增加加载动态模块支持)