258 lines
7.5 KiB
PHP
258 lines
7.5 KiB
PHP
<?php
|
||
|
||
namespace App\Services;
|
||
|
||
@error_reporting(E_ALL & ~E_NOTICE & ~E_WARNING);
|
||
|
||
use App\Models\User;
|
||
use App\Models\WebSocket;
|
||
use App\Models\WebSocketDialogMsg;
|
||
use App\Module\Base;
|
||
use App\Tasks\LineTask;
|
||
use App\Tasks\PushTask;
|
||
use Cache;
|
||
use Carbon\Carbon;
|
||
use Hhxsv5\LaravelS\Swoole\Task\Task;
|
||
use Hhxsv5\LaravelS\Swoole\WebSocketHandlerInterface;
|
||
use Swoole\Http\Request;
|
||
use Swoole\WebSocket\Frame;
|
||
use Swoole\WebSocket\Server;
|
||
|
||
/**
|
||
* @see https://wiki.swoole.com/#/start/start_ws_server
|
||
*/
|
||
class WebSocketService implements WebSocketHandlerInterface
|
||
{
|
||
/**
|
||
* 声明没有参数的构造函数
|
||
* WebSocketService constructor.
|
||
*/
|
||
public function __construct()
|
||
{
|
||
|
||
}
|
||
|
||
/**
|
||
* 连接建立时触发
|
||
* @param Server $server
|
||
* @param Request $request
|
||
*/
|
||
public function onOpen(Server $server, Request $request)
|
||
{
|
||
global $_A;
|
||
$_A = [
|
||
'__static_langdata' => [],
|
||
];
|
||
$fd = $request->fd;
|
||
$data = Base::newTrim($request->get);
|
||
$action = $data['action'];
|
||
switch ($action) {
|
||
/**
|
||
* 网页访问
|
||
*/
|
||
case 'web':
|
||
{
|
||
// 判断token参数
|
||
$token = $data['token'];
|
||
$cacheKey = "ws::token:" . md5($token);
|
||
$userid = Cache::remember($cacheKey, now()->addSeconds(1), function () use ($token) {
|
||
$authInfo = User::authFind('all', $token);
|
||
if ($authInfo['userid'] > 0) {
|
||
if (User::whereUserid($authInfo['userid'])->whereEmail($authInfo['email'])->whereEncrypt($authInfo['encrypt'])->exists()) {
|
||
return $authInfo['userid'];
|
||
}
|
||
}
|
||
return 0;
|
||
});
|
||
if (empty($userid)) {
|
||
Cache::forget($cacheKey);
|
||
$server->push($fd, Base::array2json([
|
||
'type' => 'error',
|
||
'data' => [
|
||
'error' => '会员不存在'
|
||
],
|
||
]));
|
||
$server->close($fd);
|
||
$this->deleteUser($fd);
|
||
return;
|
||
}
|
||
// 保存用户、发送open事件
|
||
$this->saveUser($fd, $userid);
|
||
$server->push($fd, Base::array2json([
|
||
'type' => 'open',
|
||
'data' => [
|
||
'fd' => $fd,
|
||
],
|
||
]));
|
||
// 通知上线
|
||
Task::deliver(new LineTask($userid, true));
|
||
// 推送离线时收到的消息
|
||
Task::deliver(new PushTask("RETRY::" . $userid));
|
||
}
|
||
break;
|
||
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 收到消息时触发
|
||
* @param Server $server
|
||
* @param Frame $frame
|
||
*/
|
||
public function onMessage(Server $server, Frame $frame)
|
||
{
|
||
global $_A;
|
||
$_A = [
|
||
'__static_langdata' => [],
|
||
];
|
||
//
|
||
$msg = Base::json2array($frame->data);
|
||
$type = $msg['type']; // 消息类型
|
||
$msgId = $msg['msgId']; // 消息ID(用于回调)
|
||
$data = $msg['data']; // 消息详情
|
||
//
|
||
$reData = [];
|
||
switch ($type) {
|
||
/**
|
||
* 收到回执
|
||
*/
|
||
case 'receipt':
|
||
return;
|
||
|
||
/**
|
||
* 已阅消息
|
||
*/
|
||
case 'readMsg':
|
||
$ids = is_array($data['id']) ? $data['id'] : [$data['id']];
|
||
$userid = $this->getUserid($frame->fd);
|
||
WebSocketDialogMsg::whereIn('id', $ids)->chunkById(20, function($list) use ($userid) {
|
||
/** @var WebSocketDialogMsg $item */
|
||
foreach ($list as $item) {
|
||
$item->readSuccess($userid);
|
||
}
|
||
});
|
||
return;
|
||
|
||
/**
|
||
* 访问状态
|
||
*/
|
||
case 'path':
|
||
$row = WebSocket::whereFd($frame->fd)->first();
|
||
if ($row) {
|
||
$pathNew = $data['path'];
|
||
$pathOld = $row->path;
|
||
$row->path = $pathNew;
|
||
$row->save();
|
||
if (preg_match("/^file\/content\/\d+$/", $pathOld)) {
|
||
$this->pushPath($pathOld);
|
||
}
|
||
if (preg_match("/^file\/content\/\d+$/", $pathNew)) {
|
||
$this->pushPath($pathNew);
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
//
|
||
if ($msgId) {
|
||
PushTask::push([
|
||
'fd' => $frame->fd,
|
||
'msg' => [
|
||
'type' => 'receipt',
|
||
'msgId' => $msgId,
|
||
'data' => $reData,
|
||
]
|
||
]);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 关闭连接时触发
|
||
* @param Server $server
|
||
* @param $fd
|
||
* @param $reactorId
|
||
* @throws \Exception
|
||
*/
|
||
public function onClose(Server $server, $fd, $reactorId)
|
||
{
|
||
Task::deliver(new LineTask($this->getUserid($fd), false)); // 通知离线
|
||
$this->deleteUser($fd);
|
||
}
|
||
|
||
/** ****************************************************************************** */
|
||
/** ****************************************************************************** */
|
||
/** ****************************************************************************** */
|
||
|
||
/**
|
||
* 保存用户
|
||
* @param $fd
|
||
* @param $userid
|
||
*/
|
||
private function saveUser($fd, $userid)
|
||
{
|
||
WebSocket::updateInsert([
|
||
'key' => md5($fd . '@' . $userid)
|
||
], [
|
||
'fd' => $fd,
|
||
'userid' => $userid,
|
||
]);
|
||
Cache::put("User::online:" . $userid, time(), Carbon::now()->addSeconds(30));
|
||
}
|
||
|
||
/**
|
||
* 清除用户
|
||
* @param $fd
|
||
*/
|
||
private function deleteUser($fd)
|
||
{
|
||
$array = [];
|
||
WebSocket::whereFd($fd)->chunk(10, function($list) use (&$array) {
|
||
/** @var WebSocket $item */
|
||
foreach ($list as $item) {
|
||
$item->delete();
|
||
if ($item->path && str_starts_with($item->path, "file/content/")) {
|
||
$array[$item->path] = $item->path;
|
||
}
|
||
}
|
||
});
|
||
foreach ($array as $path) {
|
||
$this->pushPath($path);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 根据fd获取会员ID
|
||
* @param $fd
|
||
* @return int
|
||
*/
|
||
private function getUserid($fd)
|
||
{
|
||
return intval(WebSocket::whereFd($fd)->value('userid'));
|
||
}
|
||
|
||
/**
|
||
* 发送给相同访问状态的会员
|
||
* @param $path
|
||
*/
|
||
private function pushPath($path)
|
||
{
|
||
$array = WebSocket::wherePath($path)->pluck('userid')->toArray();
|
||
if ($array) {
|
||
$userids = array_values(array_filter(array_unique($array)));
|
||
$params = [
|
||
'userid' => $userids,
|
||
'msg' => [
|
||
'type' => 'path',
|
||
'data' => [
|
||
'path' => $path,
|
||
'userids' => $userids
|
||
],
|
||
]
|
||
];
|
||
$task = new PushTask($params, false);
|
||
Task::deliver($task);
|
||
}
|
||
}
|
||
}
|