workerman + gateway +thinkphp 简单使用
1.Workerman是什么?(套用官网)
Workerman是一款纯PHP开发的开源高性能的PHP socket 服务框架。
Workerman不是重复造轮子,它不是一个MVC框架,而是一个更底层更通用的socket服务框架,你可以用它开发tcp代理、梯子代理、做游戏服务器、邮件服务器、ftp服务器、甚至开发一个php版本的redis、php版本的数据库、php版本的nginx、php版本的php-fpm等等。Workerman可以说是PHP领域的一次创新,让开发者彻底摆脱了PHP只能做WEB的束缚。
实际上Workerman类似一个PHP版本的nginx,核心也是多进程+Epoll+非阻塞IO。Workerman每个进程能维持上万并发连接。由于本身常住内存,不依赖Apache、nginx、php-fpm这些容器,拥有超高的性能。同时支持TCP、UDP、UNIXSOCKET,支持长连接,支持Websocket、HTTP、WSS、HTTPS等通讯协以及各种自定义协议。拥有定时器、异步socket客户端、异步Mysql、异步Redis、异步Http、异步消息队列等众多高性能组件。
2. GatewayWorker是什么?(套用官网)
GatewayWorker基于Workerman开发的一个项目框架,用于快速开发TCP长连接应用,例如app推送服务端、即时IM服务端、游戏服务端、物联网、智能家居等等
GatewayWorker使用经典的Gateway和Worker进程模型。Gateway进程负责维持客户端连接,并转发客户端的数据给BusinessWorker进程处理,BusinessWorker进程负责处理实际的业务逻辑(默认调用Events.php处理业务),并将结果推送给对应的客户端。Gateway服务和BusinessWorker服务可以分开部署在不同的服务器上,实现分布式集群。
3. Gatewayworker + thinkphp
数据交互模型:
流程:
- 客户端(浏览器)发出socket请求与getwayworker建立连接
- 客户端发出http请求(注意是发送http请求处理业务,所有业务都放在tp处理)
- tp处理业务逻辑(把client_id与uid绑定、客户端分组、数据库查询等),然后调用gateway的接口把结果数据进行广播
- 客户端接收广播的数据,进行视图渲染
4. 环境搭建
1. workerman 与 gateway安装
composer require workerman/workerman
composer require workerman/gateway-worker
官网实例 点击下载
2. 配置
start_gateway.php文件配置(开启gateway服务,并在注册服务注册)
<?php use \Workerman\Worker; use \Workerman\WebServer; use \GatewayWorker\Gateway; use \GatewayWorker\BusinessWorker; use \Workerman\Autoloader; // $context = array( 'ssl' => array( 'local_cert' => '/etc/pki/tls/certs/public.pem', // 或者crt文件 'local_pk' => '/etc/pki/tls/private/214498534070135.key', 'verify_peer' => false ) ); // gateway 进程,这里使用websocket协议,这里使用了443端口,所有要加载ssl的配置,websocket://0.0.0.0:443:允许所有任何客户端使用wss协议访问 $gateway = new Gateway("websocket://0.0.0.0:443",$context); $gateway->transport = 'ssl'; // gateway名称,status方便查看 $gateway->name = 'YourAppGateway'; // gateway进程数 $gateway->count = 4; // 本机ip,分布式部署时使用内网ip $gateway->lanIp = '172.31.240.231'; // 内部通讯起始端口,假如$gateway->count=4,起始端口为4000 // 则一般会使用4000 4001 4002 4003 4个端口作为内部通讯端口 $gateway->startPort = 2900; // 服务注册地址 $gateway->registerAddress = '172.31.240.231:1238'; // 心跳检测 15秒一次 $gateway->pingInterval = 15; $gateway->pingNotResponseLimit = 1; // 当pingData为空,服务器将不会向客户端发送心跳检测(为了节省服务器资源,心跳检测最好由客户端发起) $gateway->pingData = '';
3. start_businessworker.php(开启businessworker服务,并在注册服务注册)
<?php use \Workerman\Worker; use \Workerman\WebServer; use \GatewayWorker\Gateway; use \GatewayWorker\BusinessWorker; use \Workerman\Autoloader; // bussinessWorker 进程 $worker = new BusinessWorker(); // worker名称 $worker->name = 'YourAppBusinessWorker'; // bussinessWorker进程数量 $worker->count = 4; // 服务注册地址 $worker->registerAddress = '172.40.239.231:1238'; // 如果不是在根目录启动,则运行runAll方法 if(!defined('GLOBAL_START')) { Worker::runAll(); }
4. start_register.php(开启注册服务)
<?php use \Workerman\Worker; use \GatewayWorker\Register; // register 服务必须是text协议 $register = new Register('text://172.40.239.231:1238'); // 如果不是在根目录启动,则运行runAll方法 if(!defined('GLOBAL_START')) { Worker::runAll(); }
5. 启动服务
启动
以debug(调试)方式启动
php start.php start
以daemon(守护进程)方式启动
php start.php start -d
停止
php start.php stop
重启
php start.php restart
平滑重启
php start.php reload
查看状态
php start.php status
5. 客户端操作(这是我基于小程序接口,封装的一个socket库)
var socket = new Socket('wss://wss.xinyuruiyang.com'); socket.on("open",function (res) { console.log('WebSocket连接已打开!') _this.setData({ isOpenSocket: true }); });
6. gateway端操作
<?php use \GatewayWorker\Lib\Gateway; use \Workerman\Lib\Timer; class Events { public static $worker_id = null; /** * 当客户端连接时触发 * 如果业务不需此回调可以删除onConnect * * @param int $client_id 连接id */ public static function onConnect($client_id) { // 向当前client_id发送数据(触发客户端的init时间) Gateway::sendToClient($client_id, json_encode(["init",["client_id"=>$client_id]])); } }
7. tp端操作
private function bind_user($room,$client_id,$uid=''){ if( empty($uid) ){ $sk = $this->checksession(); $openid = $this->get_openid($sk); $uid = D('User')->where(['openid'=>$openid])->getField('id'); } // client_id与uid绑定 Gateway::bindUid($client_id, $uid); // 加入某个群组(可调用多次加入多个群组) Gateway::joinGroup($client_id, $room);
// 设置session Gateway::setSession($client_id, ['uid'=>$uid]);
// 获取客户端分组人数 $number = Gateway::getClientCountByGroup($room); D('Match')->where(['rid'=>$room,'uid'=>$uid,'status'=>['in',['1','2']]])->save(['client_id'=>$client_id]); $message = D('Match')->where(['rid'=>$room,'status'=>['in',['1','2']]])->select(); foreach($message as &$v){ $v['user'] = D('User')->where(['id'=>$v['uid']])->find(); $v['user']['teamside'] = D('Match')->where(['rid'=>$room,'uid'=>$v['uid'],'status'=>['in',['1','2']]])->getField('teamside'); }
// 向某个房间广播数据 Gateway::sendToGroup($room, json_encode(["room",$message])); }
}