当前位置: 首页 > web >正文

在 Laravel 12 中实现 WebSocket 通信时进行身份验证

在 Laravel 12 中实现 WebSocket 通信时,若需在身份验证失败后主动断开客户端连接,需结合 频道认证机制服务端主动断连操作。以下是具体实现步骤:


一、身份验证流程设计

WebSocket 连接的身份验证通常通过 私有频道(Private Channel)存在频道(Presence Channel) 实现。以下是核心流程:

  1. 客户端发起订阅请求
    // 前端订阅私有频道
    Echo.private('chat.1').listen('NewMessage', (data) => { /* ... */ });
    
  2. 服务端验证权限
    • Laravel 自动触发 routes/channels.php 中的频道授权逻辑。
    • 若验证失败,返回 403 Forbidden,并触发断连操作。

二、实现身份验证失败断连

方案 1:使用 Laravel Reverb(官方方案)

Reverb 默认在频道认证失败时拒绝订阅,但不会主动断开连接。需通过以下方式强制断连:

  1. 自定义认证中间件

    • 创建中间件 VerifyWebSocketAuth
      php artisan make:middleware VerifyWebSocketAuth
      
    • 编辑中间件逻辑:
      <?php
      namespace App\Http\Middleware;
      use Closure;
      use Illuminate\Support\Facades\Auth;class VerifyWebSocketAuth {public function handle($request, Closure $next) {if (!Auth::check()) { // 身份验证失败// 返回 403 并关闭连接abort(403, 'Unauthorized');}return $next($request);}
      }
      
  2. 将中间件应用到广播路由

    • 修改 app/Providers/BroadcastServiceProvider.php
      public function boot() {Broadcast::routes(['middleware' => ['web', 'auth:api', VerifyWebSocketAuth::class]]);
      }
      
  3. 前端处理断连

    Echo.private('chat.1').listen('NewMessage', (data) => { /* ... */ }).error((error) => {console.error('连接被拒绝:', error);Echo.disconnect(); // 主动断开连接});
    

方案 2:使用 Soketi(Pusher 替代)

Soketi 作为 Pusher 协议兼容服务,可通过 自定义授权端点 实现强制断连:

  1. 配置自定义授权端点
    • .env 中指定授权路由:
      PUSHER_APP_AUTHORIZER=http://your-app.test/broadcasting/auth
      
  2. 自定义授权响应
    • routes/api.php 中定义严格认证逻辑:
      use Illuminate\Support\Facades\Broadcast;Broadcast::channel('chat.{userId}', function ($user, $userId) {if ($user->id !== (int) $userId) {// 返回 403 并附带断连指令return response()->json(['error' => 'Forbidden'], 403)->header('Connection', 'close');}return true;
      });
      

方案 3:使用 Workerman/Swoole(底层控制)

若需直接操作 WebSocket 连接,可通过 Workerman/Swoole 的底层 API 强制断连:

  1. 保存客户端连接标识
    • 在连接建立时记录客户端 ID(如 fd):
      $worker->onConnect = function ($connection) {$connection->id = uniqid();Cache::put('ws_connections.' . $connection->id, $connection);
      };
      
  2. 认证失败时主动断连
    $worker->onMessage = function ($connection, $data) {$user = Auth::user();if (!$user) {// 发送错误消息并关闭连接$connection->send(json_encode(['error' => 'Unauthorized']));$connection->close();return;}// 正常处理消息
    };
    

三、生产环境优化

  1. 日志记录
    // 在中间件或事件中记录断连行为
    Log::warning('WebSocket 连接因认证失败被关闭', ['ip' => request()->ip(),'time' => now()
    ]);
    
  2. 速率限制
    • app/Providers/RouteServiceProvider.php 中限制认证请求频率:
      RateLimiter::for('websocket-auth', function (Request $request) {return Limit::perMinute(30);
      });
      

四、完整示例代码

服务端(Laravel Reverb)
  1. 频道授权文件 (routes/channels.php):
    Broadcast::channel('chat.{userId}', function ($user, $userId) {if ($user->id !== (int) $userId) {// 抛出异常触发断连throw new \Exception('身份验证失败');}return true;
    });
    
  2. 全局异常处理 (app/Exceptions/Handler.php):
    public function register() {$this->renderable(function (\Exception $e, $request) {if ($e->getMessage() === '身份验证失败') {return response()->json(['error' => 'Forbidden'], 403)->header('Connection', 'close');}});
    }
    

客户端(Laravel Echo)
// 监听连接错误
Echo.connector.pusher.connection.bind('error', (error) => {if (error.status === 403) {Echo.disconnect();alert('身份验证失败,连接已关闭');}
});

五、关键注意事项

  1. 协议兼容性
    • 使用 wss:// 协议时确保 SSL 证书有效。
  2. 连接状态同步
    • 使用 Redis 或数据库跟踪在线用户状态。
  3. 测试工具
    • 使用 WebSocket KingPostman 手动测试断连逻辑。

通过以上方案,可以在 Laravel 12 中实现 WebSocket 通信时的严格身份验证,确保验证失败时主动断开连接,提升系统安全性。

http://www.xdnf.cn/news/4127.html

相关文章:

  • ts bug 找不到模块或相应类型的声明,@符有红色波浪线
  • Prometheus实战教程:k8s平台-使用文件服务发现案例
  • Android Retrofit框架分析(三):自动切换回主线程;bulid的过程;create方法+ServiceMethod源码了解
  • 【Azure Redis 缓存】关于Azure Cache for Redis 服务在传输和存储键值对(Key/Value)的加密问题
  • Windows系统修改Docker Desktop(WSL2)内存分配
  • Facebook隐私保护措施的优缺点解析
  • Java面试全栈解析:Spring Boot、Kafka与Redis实战揭秘
  • Jenkins+Newman实现接口自动化测试
  • 蓝桥杯-通电(最小生成树java)
  • Axure : 列表分页、 列表翻页
  • 第1.3讲、什么是 Attention?——从点菜说起 [特殊字符]️
  • FastJSON 使用 `Feature.OrderedField` 修复 `JSONObject` 序列化字段顺序问题
  • 用 GRPO 魔法点亮Text2SQL 的推理之路:让模型“思考”得更像人类
  • AI服务器的作用都有哪些?
  • 【工具使用-数据可视化工具】Apache Superset
  • Cursor 被封解决方案
  • 2、Kafka Replica机制与ISR、HW、LEO、AR、OSR详解
  • .NET 通过回调函数执行 Shellcode启动进程
  • 广州华锐视点邀您参与2025广交会VRAR展【5月10-12日】
  • 快速体验 .NET9 提供的 HybridCache 混合缓存
  • wrod生成pdf。[特殊字符]改背景
  • 基于Piecewise Jerk Speed Optimizer的速度规划算法(附ROS C++/Python仿真)
  • C++多态详解
  • ORCAD打印pdf
  • Docker手动重构Nginx镜像,融入Lua、Redis功能
  • 【C++】WSL常用语法
  • 先滤波再降采样 还是 先降采样再滤波
  • IL2CPP 技术深度解析
  • std::move()详解
  • n8n 使用 Merge 节点进行数据聚合