layim和Gatewayworker组合的实时通讯
今天是第二次重新开发使用layim和Gatewayworker,但是由于第一次没有写文档,导致这一次就跟第一次一样,一头雾水,重新开始看文档研究,导致遇到一个瓶颈,怎么都过不去。所以,以这篇文章开始,以后所有的知识点,全部写上文档。
首先如果是快速入手的快,直接从http://www.workerman.net/windows下载,worker。然后解压放到项目中去。
,除了以上这三个文件,其他文件不需要做任何修改。
1 Events.php
这里面内容将是操控数据库和实现页面内容返回的地方,所以可以在这里面自定义任何方法。

<?php /** * This file is part of workerman. * * Licensed under The MIT License * For full copyright and license information, please see the MIT-LICENSE.txt * Redistributions of files must retain the above copyright notice. * * @author walkor<walkor@workerman.net> * @copyright walkor<walkor@workerman.net> * @link http://www.workerman.net/ * @license http://www.opensource.org/licenses/mit-license.php MIT License */ /** * 用于检测业务代码死循环或者长时间阻塞等问题 * 如果发现业务卡死,可以将下面declare打开(去掉//注释),并执行php start.php reload * 然后观察一段时间workerman.log看是否有process_timeout异常 */ //declare(ticks=1); // 自动加载类 use \GatewayWorker\Lib\Gateway; use \GatewayWorker\Lib\DbConnection; /** * 主逻辑 * 主要是处理 onConnect onMessage onClose 三个方法 * onConnect 和 onClose 如果不需要可以不用实现并删除 */ class Events { public static $db; public static $uuid = []; public static $clientid = []; public static function onWorkerStart() { self::$db = new DbConnection('xxx', '3306', 'xxx', xxx', 'xxx'); } /** * 当客户端连接时触发 * 如果业务不需此回调可以删除onConnect * * @param int $client_id 连接id */ public static function onConnect($client_id){ // Gateway::sendToClient($client_id, json_encode($noonline)); } /** * 当客户端发来消息时触发 * @param int $client_id 连接id * @param mixed $message 具体消息 */ public static function onMessage($client_id, $message) { $data = json_decode($message,true); switch($data['type']) { //当有用户上线 case 'reg': //绑定uid 用于数据分发 $uid = $data['data']['id']; Gateway::bindUid($client_id, $uid); self::$clientid[$client_id] = $uid; //如果数组中不存在键值uid,那么表示是第一次登陆,更新数据库 if(!isset(self::$uuid[$uid])){ self::$uuid[$uid] = $uid; self::$db->update('im_user_baseInfo')->cols(array('status'=>1))->where('id='.$uid)->query(); } break; case 'chatMessage': //正常聊天 $to_user = $data['data']['to']['id']; $from_user = $data['data']['mine']['id']; $time = time(); $content = quotationMarks($data['data']['mine']['content']); $msg = array( 'username'=> $data['data']['mine']['username'],//消息来源用户名 'avatar'=> $data['data']['mine']['avatar'], 'id'=> $from_user,//消息的来源ID(如果是私聊,则是用户id,如果是群聊,则是群组id) 'content'=> $content, 'type'=> 'friend',//聊天窗口来源类型,从发送消息传递的to里面获取 'mine'=> false,//是否我发送的消息,如果为true,则会显示在右方 'timestamp'=> date('Y-m-d h:i:s',$time), 'emit'=> 'chatMessage' ); //发送给自己的,同客户端同步 $msg2 = array( 'username'=> $data['data']['mine']['username'],//消息来源用户名 'avatar'=> $data['data']['mine']['avatar'], 'id'=> $to_user,//消息的来源ID(如果是私聊,则是用户id,如果是群聊,则是群组id) 'content'=> $content, 'type'=> 'friend',//聊天窗口来源类型,从发送消息传递的to里面获取 'mine'=> true,//是否我发送的消息,如果为true,则会显示在右方 'timestamp'=> date('Y-m-d h:i:s',$time), 'emit'=> 'sendme' ); $sql = array( 'from_user'=>$from_user, 'to_user'=>$to_user, 'type'=>1,//目前默认都是朋友 'content'=>$content, 'msg_type'=>1,//1:文本 2:资源(图片,文件等) 'add_time'=>$time, 'is_read'=>1,//已经查看 ); //判断是否在线 if (Gateway::isUidOnline($to_user)) { Gateway::sendToUid($to_user, json_encode($msg)); self::$db->insert('im_message')->cols($sql)->query(); } else { //如果对方离线,发送系统消息 $sql['is_read'] = 0; self::$db->insert('im_message')->cols($sql)->query(); $noonline = array( 'emit'=>'status', 'fromid'=>-2, 'id'=>-2, 'status'=>0, ); Gateway::sendToClient($client_id, json_encode($noonline)); } //根据uid获取其他客户端的client_id $from_client_list = Gateway::getClientIdByUid($from_user); unset($from_client_list[array_search($client_id,$from_client_list)]); foreach ($from_client_list as $from_client){ Gateway::sendToClient($from_client, json_encode($msg2)); } break; //查询是否在线的状态 case 'status': $to_user = $data['data']['id']; $noonline = array( 'emit'=>'status', 'fromid'=>-2, 'id'=>-2, 'status'=>0 ); if (Gateway::isUidOnline($to_user)) { $noonline['status'] = 1; } Gateway::sendToClient($client_id, json_encode($noonline)); break; //修改聊天记录为已读状态 case 'isread': $username = quotationMarks($data['data']['name']); $uid = quotationMarks($data['data']['mid']); $from_user = self::getUserinfoByname($username); //改变消息的已读状态 if(!empty($from_user) && !empty($uid)) self::changeIsread($from_user,$uid,1); break; } } /** * 当用户断开连接时触发 * @param int $client_id 连接id */ public static function onClose($client_id) { $id = self::$clientid[$client_id]; unset(self::$clientid[$client_id]); //如果长度等于0,说明绑定该uid的所有客户端都下线了 if(count(Gateway::getClientIdByUid($id)) == 0){ unset(self::$uuid[$id]); self::$db->update('im_user_baseInfo')->cols(array('status'=>0))->where('id='.$id)->query(); } // var_export(self::$clientid); } /** * 查询该用户名对应的用户 * @param $username * @return mixed */ public static function getUserinfoByname($username){ $result = self::$db->select('id')->from('im_user_baseInfo')->where("username= '$username' ")->column(); return $result[0]; } /** * Mobile将消息设置为已读状态 * @param $from_user * @param $to_user * @param $is_read */ public static function changeIsread($from_user,$to_user,$is_read){ $sql = "UPDATE im_message SET is_read = $is_read WHERE from_user = $from_user AND to_user = $to_user"; self::$db->query($sql); } } /** * 数据接收正则匹配,去掉字符串中的单引号和双引号 * add by fpc time:2017-03-29 */ function quotationMarks($str){ // 正则表达式匹配中文(UTF8编码) if(preg_match_all('/\'(.*?)\'/',$str)){ $str = str_replace("\'",'',$str); if(preg_match_all('/\"(.*?)\"/',$str)){ $str = str_replace('\"','',$str); } return trim($str); }else{ return addslashes(trim($str)); } }
2start_businessworker.php
3start_gateway.php
4客户端
总共就以上这些代码,以上代码仅在windows下的代码,今天遇到一个问题,由于文件和方法是从linux移动过来的,所以调试的时候就出现了一下问题
服务器一直是可以的,但是使用
var socket = new WebSocket('ws://192.168.0.188:8484');打开的时候,就一直连接不上,原因就在于在start_gateway.phpz这个文件中,
$gateway = new Gateway("websocket://0.0.0.0:8484");这一句,linux是 tcp://.......... windows是websocket。
接下来就是无限发挥的时刻了
如果是https的服务器,那么使用
var socket = new WebSocket('wss://192.168.0.188:8484');
第二部分
摘要:在一个服务器上可以有多个 WebSocket部署
说明:第二个使用的是https
当我在同一个服务器上,使用workman部署第二个websocket的时候如图所示为了防止端口号冲突,要修改端口号与以第一个部署的服务器的端口号全部不一样
因为使用了https,与http的区别就是增加了这两层代码
在前台的js中
var socket = new WebSocket('wss://www.xxx.com:7272'); wss://域名+端口号,端口号要注意和gate的一致