在 Laravel 中使用 PHPSocketIO 提供服务
时间:2024-1-11 21:58 作者:wanzi 分类: php
参考测试文章,https://www.workerman.net/a/1621 可以了解到通过 laraman 与 phpsocketio 结合的使用方式。接下来便可以通过自定义进程为项目提供即时通讯服务。下方是部分 demo 代码。
config/socketio.php
<?php
use Itinysun\Laraman\Events\MessageDone;
use Itinysun\Laraman\Events\MessageReceived;
use Itinysun\Laraman\Events\RequestReceived;
use Itinysun\Laraman\Events\TaskReceived;
use Itinysun\Laraman\Listeners\CleanBaseState;
use Itinysun\Laraman\Listeners\CleanWebState;
use Itinysun\Laraman\Process\Web;
/*
* 这里是进程配置文件
* 配置文件格式:
* 'process name'=>[
* 'handler'=>class name, 必须,写进程类名。必须继承 ProcessBase 类
* 'options'= [ 进程类构造函数的参数,用于进程内部使用
* 'events'=>[event name =>array(listeners)] 可选,订阅事件。
* 'clearMode'=>bool , 可选,默认false,是否开启洁癖模式
* ],
* 'workerman'=>array config 可选,构造 worker时的参数,参考workerman官方手册,如果是全局属性,请在server中配置
* ]
*
*/
return [
'workerman' => [
'count' => 1,
//是否支持平滑重启,如果是持续性任务,请设置为false
'reloadable' => true,
//运行命令使用的用户及组,windows无效
'user' => '',
'group' => '',
],
'handler' => \App\Process\SocketIO::class,
];
app/Process/SocketIO.php
<?php
namespace App\Process;
use Workerman\Timer;
use Workerman\Worker;
use Workerman\Protocols\Http\Request;
use Workerman\Connection\TcpConnection;
use Itinysun\Laraman\Process\ProcessBase;
use Itinysun\Laraman\Server\LaramanWorker;
use Itinysun\Laraman\Traits\HasLaravelApplication;
class SocketIO extends ProcessBase
{
use HasLaravelApplication;
protected function onWorkerStart(LaramanWorker $worker): void
{
// 全局数组保存uid在线数据
$uidConnectionMap = array();
// 记录最后一次广播的在线用户数
$last_online_count = 0;
// 记录最后一次广播的在线页面数
$last_online_page_count = 0;
// PHPSocketIO服务
$sender_io = new \PHPSocketIO\SocketIO(2120);
// 客户端发起连接事件时,设置连接socket的各种事件回调
$sender_io->on('connection', function ($sender_io) use (&$uidConnectionMap, $last_online_count, $last_online_page_count) {
// 当客户端发来登录事件时触发
$sender_io->on('login', function ($uid) use ($sender_io, &$uidConnectionMap, $last_online_count, $last_online_page_count) {
// 已经登录过了
if (isset($sender_io->uid)) {
return;
}
// 更新对应uid的在线数据
$uid = (string)$uid;
if (!isset($uidConnectionMap[$uid])) {
$uidConnectionMap[$uid] = 0;
}
// 这个uid有++$uidConnectionMap[$uid]个socket连接
++$uidConnectionMap[$uid];
// 将这个连接加入到uid分组,方便针对uid推送数据
$sender_io->join($uid);
$sender_io->uid = $uid;
// 更新这个socket对应页面的在线数据
$sender_io->emit('update_online_count', "当前<b>{$last_online_count}</b>人在线,共打开<b>{$last_online_page_count}</b>个页面");
});
// 当客户端断开连接是触发(一般是关闭网页或者跳转刷新导致)
$sender_io->on('disconnect', function () use ($sender_io, &$uidConnectionMap) {
if (!isset($sender_io->uid)) {
return;
}
if (empty($uidConnectionMap[$sender_io->uid])) {
return;
}
// 将uid的在线socket数减一
if (--$uidConnectionMap[$sender_io->uid] <= 0) {
unset($uidConnectionMap[$sender_io->uid]);
}
});
});
// 监听一个http端口
$inner_http_worker = new Worker('http://0.0.0.0:2121');
// 当http客户端发来数据时触发
$inner_http_worker->onMessage = function (TcpConnection $http_connection, Request $request) use ($sender_io, &$uidConnectionMap) {
$response = new \Workerman\Protocols\Http\Response();
$response->withHeaders([
'Access-Control-Allow-Credentials' => 'true',
'Access-Control-Allow-Origin' => $request->header('origin', '*'),
'Access-Control-Allow-Methods' => $request->header('access-control-request-method', '*'),
'Access-Control-Allow-Headers' => $request->header('access-control-request-headers', '*'),
'Content-Type' => 'application/json',
]);
$post = $request->post();
$post = $post ? $post : $request->get();
// 推送数据的url格式 type=publish&to=uid&content=xxxx
switch (@$post['type']) {
case 'publish':
$to = @$post['to'];
$post['content'] = htmlspecialchars(@$post['content']);
// 有指定uid则向uid所在socket组发送数据
if ($to) {
$sender_io->to($to)->emit('new_msg', $post['content']);
// 否则向所有uid推送数据
} else {
$sender_io->emit('new_msg', @$post['content']);
}
// http接口返回,如果用户离线socket返回fail
if ($to && !isset($uidConnectionMap[$to])) {
$response->withBody('offline');
return $http_connection->send($response);
} else {
$response->withBody('ok');
return $http_connection->send($response);
}
}
$response->withBody('fail');
return $http_connection->send($response);
};
// 执行监听
$inner_http_worker->listen();
$sender_io->worker->listen();
// 一个定时器,定时向所有uid推送当前uid在线数及在线页面数
Timer::add(1, function () use (&$uidConnectionMap, $sender_io, &$last_online_count, &$last_online_page_count) {
$online_count_now = count($uidConnectionMap);
$online_page_count_now = array_sum($uidConnectionMap);
// 只有在客户端在线数变化了才广播,减少不必要的客户端通讯
if ($last_online_count != $online_count_now || $last_online_page_count != $online_page_count_now) {
$sender_io->emit('update_online_count', "当前<b>{$online_count_now}</b>人在线,共打开<b>{$online_page_count_now}</b>个页面");
$last_online_count = $online_count_now;
$last_online_page_count = $online_page_count_now;
}
});
}
}