Laravel/Lumen 中使用 Echo + Socket.IO-Client 实现网页即时通讯广播

此处以 Lumen 9 框架为例说明如何调试通过 Echo 服务端以及客户端

安装

Redis

composer require illuminate/redis,如果安装失败需要根据当前框架版本指定所需 Redis 版本,例如:composer require illuminate/redis "^9.0"

Broadcasting

同 Redis,composer require illuminate/broadcasting "^9.0"

laravel-echo-server

通过 yarnnpm 全局安装 laravel-echo-server,如:yarn global add laravel-echo-server

注意:需要把路径 C:\Users\用户名\AppData\Local\Yarn\bin 加入环境变量 Path

在 macOS 中如果未修改过 /etc/paths 文件,你可能需要把 /usr/local/bin 加入到环境变量 Path 中
关于 /etc/paths 文件,可以参考这篇文章:https://maxsky.blog.csdn.net/article/details/53930406
注意 .bash_profile 文件现在变成了 .zshrc

此时运行 laravel-echo-server --version 即可看到版本号

配置

Redis 配置

bootstrap/app.php 中确保 $app->register(App\Providers\AppServiceProvider::class); 非注释 状态

然后打开 app/Providers/AppServiceProvider.php 文件,在 register 方法中添加如下内容:

use Illuminate\Redis\RedisServiceProvider;

// ...
public function register(): void {
    $this->app->register(RedisServiceProvider::class);
}

Broadcast 配置

同 Redis 需要在 AppServiceProvider 中新增 Provider,现在看起来就像这样:

use Illuminate\Broadcasting\BroadcastServiceProvider;
use Illuminate\Redis\RedisServiceProvider;

// ...
public function register(): void {
    $this->app->register(BroadcastServiceProvider::class);
    $this->app->register(RedisServiceProvider::class);
}

Lumen 框架用户需要在 bootstrap/app.php 中的 $app->configure() 下新增一条 $app->configure('broadcasting');,使配置文件生效

然后在 $app->register(App\Providers\AppServiceProvider::class); 下新增:

// 稍后创建该 Provider
$app->register(App\Providers\BroadcastServiceProvider::class);

最后复制 vendor/laravel/lumen-framework/config/broadcasting.php 文件至 config 目录下

记得配置 Redis 驱动,随后在项目根目录的 .env 文件中新增

BROADCAST_DRIVER=redis

Broadcast 使用 Redis 的 default 连接配置,所以在 Redis 中的 db 号是 REDIS_DB,默认为 0

配置 Facade

打开 bootstrap/app.php 文件,确保 $app->withFacades();非注释 状态,将该句修改为如下内容:

// ...
// 注意下方两句需在最后的 $app->configure() 方法之后

$app->withFacades(true, array_flip(config('app.aliases')));

$app->withEloquent();
// ...

打开 config 目录下存在 app.php,在最下方新增 aliases 部分,看起来就像:

return [
    'name' => env('APP_NAME', 'Lumen'),

    // ...

    'key' => env('APP_KEY'),

    'cipher' => 'AES-256-CBC',

    'aliases' => [
        'App'       => Illuminate\Support\Facades\App::class,
        'Broadcast' => Illuminate\Support\Facades\Broadcast::class,
        'Eloquent'  => Illuminate\Database\Eloquent\Model::class,
        'File'      => Illuminate\Support\Facades\File::class,
        'Hash'      => Illuminate\Support\Facades\Hash::class,
        'Http'      => Illuminate\Support\Facades\Http::class,
        'View'      => Illuminate\Support\Facades\View::class,
    ]
];

注意低版本的 Lumen 框架中可能不存在 Http 等 Facade

新增 BroadcastServiceProvider

app/Providers 下新增 BroadcastServiceProvider.php 文件

<?php

namespace App\Providers;

use Broadcast;
use Illuminate\Support\ServiceProvider;

class BroadcastServiceProvider extends ServiceProvider {

    /**
     * Bootstrap any application services.
     *
     * @return void
     */
    public function boot(): void {
        // 频道授权回调,闭包中的第一个参数为已认证用户身份模型
        // 用户认证相关内容下方有简述,具体可参考(注意文档为 9.x 版本):https://learnku.com/docs/laravel/9.x/authentication/12239
        // ---
        // channel 第一个参数为频道名称,此处以私有频道举例所以关联了一个 ID。该 ID 与闭包函数中的 $user_id 为绑定关系
        Broadcast::channel('test.{user_id}', function ($user, int $user_id) {
            return $user->id === $user_id;
        });

        $this->app['router']->group([
            'middleware' => []
        ], function (/** @noinspection PhpUnusedParameterInspection */ $router) {
            require_once base_path('routes/channels.php');
        });
    }
}
用户认证简述

此处仅针对 Lumen 框架举例说明,且用户认证方式为 Token(如 JWT)

用户模型(Model)中需实现 AuthorizableAuthenticatable 并引入 AuthenticatableAuthorizable 两个 trait

use Illuminate\Auth\Authenticatable;
use Illuminate\Contracts\Auth\Access\Authorizable as AuthorizableContract;
use Illuminate\Contracts\Auth\Authenticatable as AuthenticatableContract;
use Laravel\Lumen\Auth\Authorizable;

class User extends Model implements AuthenticatableContract, AuthorizableContract {
    
    use Authenticatable, Authorizable;

    protected $table = 'user_table';

    protected $guarded = [];

    protected $hidden = [
        'password',
    ];
}

随后修改 app/Providers/AuthServiceProvider.php,在 boot 方法中实现自己的 Token 认证逻辑即可:

/**
 * Boot the authentication services for the application.
 *
 * @return void
 */
public function boot(): void {
    $this->app['auth']->viaRequest('api', function (Request $request) {
        // 如果是自定义的 Header,通过 $request->header('Custom-Header') 获取即可
        $token = $request->bearerToken();

        if ($token) {
            // TODO: 自定义验证逻辑

            if ($verified) {
                 return User::find('Verified User Identity'); // 如 user_no、user_id 等,此处返回用户模型
            }
        }

        return null;
    });
}

laravel-echo-server 初始化

在项目根目录运行 laravel-echo-server init,得到如下提示(参考问号后的内容):

Do you want to run this server in development mode? Yes
Which port would you like to serve from? 6001
Which database would you like to use to store presence channel members? redis
Enter the host of your Laravel authentication server. http://localhost
Will you be serving on http or https? http                  
Do you want to generate a client ID/Key for HTTP API? No   
Do you want to setup cross domain access to the API? No
What do you want this config to be saved as? laravel-echo-server.json

Configuration file saved. Run laravel-echo-server start to run server.

laravel-echo-server.json 文件详细说明及配置见:

  1. 配置选项
  2. 数据库配置
  3. Redis 可选项
  4. Socket.IO 可选项

此处我们只需要确保 authHost 对应上我们项目地址,其它配置保持默认即可。authHost 可以是 IP+端口的形式,如:http://127.0.0.1:8888

发送广播

编写相关文件

首先在 routes 下新建 channels.php 文件,内容如下:

<?php

// 稍后创建该控制器
use App\Http\Controllers\Broadcast\BroadcastAuthController;

/** @var Laravel\Lumen\Routing\Router $router */
$router->addRoute(['GET', 'POST'], 'broadcasting/auth', BroadcastAuthController::class . '@authenticate');

// 该路由用于模拟推送通知
$router->get('notice', function () {
    Broadcast::event((new App\Events\ExampleBroadcastEvent(22, '你好啊')));
});

接着在 app/Http/Controllers 下新建 Broadcast 目录,然后在里面新建 BroadcastAuthController 控制器,内容如下:

<?php

namespace App\Http\Controllers\Broadcast;

use Illuminate\Http\Request;
use Illuminate\Support\Facades\Broadcast;
use Laravel\Lumen\Routing\Controller;

class BroadcastAuthController extends Controller {

    public function authenticate(Request $request) {
        if ($request->hasSession()) {
            $request->session()->reflash();
        }

        // 注意这里我参照框架提供的用户认证实现授权,如果你有自己的一套认证代码可不使用
        return Broadcast::auth($request);
    }
}

最后创建 ExampleBroadcastEvent,生产者部分就完成了。进入 app/Events,新建 ExampleBroadcastEvent.php,内容如下:

<?php

namespace App\Events;

use Illuminate\Broadcasting\PrivateChannel;
use Illuminate\Contracts\Broadcasting\ShouldBroadcast;

class ExampleBroadcastEvent extends Event implements ShouldBroadcast {

    /** @var string broadcast queue name */
    public string $queue = 'broadcast'; // 指定队列名,默认为 default

    private int    $user_id;
    private string $message;

    /**
     * Create a new event instance.
     *
     * @return void
     */
    public function __construct(?int $user_id = null, string $message = 'Hello') {
        $this->user_id = $user_id;
        $this->message = $message;
    }

    /**
     * Set the channel to be broadcast.
     *
     * @return array
     */
    public function broadcastOn(): array {
        // 此处使用私人频道,此处的频道名必须和频道授权回调中一致
        return [new PrivateChannel("test.$this->user_id")];
    }

    /**
     * Use this method to set the broadcast name, the current class name is used by default.
     */
    public function broadcastAs(): string {
        // 广播名称,也可以叫做事件名称,此处为指定一个别名。如果不指定该方法,默认广播名为 App\Events\ExampleBroadcastEvent
        return 'ExampleNotice';
    }

    /**
     * Set the broadcast payload.
     *
     * @return array
     */
    public function broadcastWith(): array {
        // 需通过 Socket 传输到 Web 端的数据
        return [
            'data' => [
                'user_id' => $this->user_id,
                'message' => $this->message
            ]
        ];
    }
}

测试

前面我们在 channels.php 中编写了一个路由 /test_push,此时我们可以请求该路由瞧瞧广播是否已进入 Redis 队列

请求一次即可,然后使用 RDM 等工具查看是否已有数据:
广播队列

消费广播

运行服务端

项目根目录运行:

laravel-echo-server start

得到如下提示即为正确:

L A R A V E L  E C H O  S E R V E R

version 1.6.3                      

⚠ Starting server in DEV mode...   

✔  Running at localhost on port 6001
✔  Channels are ready.          
✔  Listening for http events... 
✔  Listening for redis events...

Server ready!

准备客户端

参考:https://github.com/maxsky/Laravel-Echo-Client-Demo

该 Demo 需要编辑 js/app.js 文件,根据项目情况生成 Token,可写死在 auth.headers 中或使用 bearerToken

因为我们这里使用了私人频道,所以 channel() 方法应该修改为 private()

Channel_Name 需修改为 test. 加上 Token 所对应的用户 ID,如 test.22

ExampleEvent 需修改为 ExampleNotice。如果你删掉了 ExampleBroadcastEvent 中的 broadcastAs() 方法,此处应当输入 ExampleBroadcastEvent 并移除上分 namespace 配置项或将 null 修改为 'App.Events'

这部分代码修改后看起来就像这样:

window.Echo = new Echo({
    broadcaster: 'socket.io',
    host: 'http://127.0.0.1:6001',
    auth: {
        headers: {
            'User-Agent': navigator.userAgent,
            'Custom-Header': 'Custom Header Value',
        }
    },
    withCredentials: true,
    bearerToken: 'Your Token',
    namespace: null // only for used `broadcastAs` method
});

window.Echo.private('test.22').listen('ExampleNotice', (res) => {
    console.log(res);
});

运行客户端

在上述 Demo 中,运行 yarn dev 后默认可通过 http://127.0.0.1:5173 访问客户端

注意页面没内容,我们需要按下 F12 打开开发者工具并切换到 Network/网络 选项卡进行观察

可以刷新页面,找到一条类似 ws://127.0.0.1:6001/socket.io/?EIO=3&transport=websocket&sid=XXXXX 的链接,选中后点击右侧的 Messages/消息 选项卡等待即可

运行队列

在项目根目录下运行:

php artisan queue:listen

如果出现处理提示说明消费成功。当然,因为我们在 ExampleBroadcastEvent.php 文件中修改了队列名所以此处不会出现任何执行信息

我们需在该命令中添加参数 --queue=队列名,所以 Ctrl+C 终止运行后试试这句:php artisan queue:listen --queue=broadcast

如果没有任何问题,应该会看到类似下面的提示:

2023-05-01 05:00:01 App\Events\ExampleBroadcastEvent ........ RUNNING
2023-05-01 05:00:01 App\Events\ExampleBroadcastEvent .... 7.80ms DONE

转过头再看看 laravel-echo-server 那边,除了会出现之前刷新时的 joined channel: test 外,应该会有下方提示:

Channel: private-test.22
Event: example
Channel: private-test.22
Event: example

如果一切顺利,回到浏览器,查看刚刚的 Messages/消息 中,是否出现了 ExampleBroadcastEventbroadcastWith() 返回的内容

最后

  • Web 端 socket.io-client 客户端必须使用 ^2.0 版本,不可使用 ^3.0^4.0(Client Demo 中已说明),除非 laravel-echo-server 中依赖的 socket.io 服务版本变更。所以 laravel-echo-server.json 配置中 socketio.allowEIO3 选项是无效的。具体兼容说明见:https://socket.io/docs/v4/troubleshooting-connection-issues#the-client-is-not-compatible-with-the-version-of-the-server
  • 部分配置项可写在项目根目录的 .env 环境文件中,支持的环境变量见:envVariables

流程总结

  1. 启动 WebSocket 服务端,在项目根目录运行 laravel-echo-server start
  2. 编写广播事件,参考 ExampleBroadcastEvent
  3. 通过 Broadcast::event() 方法生产消息并存储于 Redis 中等待队列消费,例如:Broadcast::event((new App\Events\ExampleBroadcastEvent(22, 'Hello')))
  4. 启动 Web 网页并连接到服务端,连接端口为 laravel-echo-server.json 中的 port 配置项;
  5. 使用命令启动队列监听以执行消费,根据示例文件应当在项目根目录运行 php artisan queue:listen --queue=broadcast
  6. 队列提示消费 DONE 完成后,事件中 broadcastWith 方法的返回值将被传送至 Web 端,至此流程结束

频道授权需仔细参考框架文档用户认证以及授权频道部分

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/536488.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

室外超声波自动气象站设备

TH-CQX10随着科技的进步和气象学的发展&#xff0c;气象监测设备已经从传统的有人值守模式转变为自动化、智能化的无人值守模式。室外超声波自动气象站设备就是这一转变的杰出代表。以下是室外超声波自动气象站设备的原理、应用及其优势&#xff1a; 1、室外超声波自动气象站设…

使用 Axios 处理 AxiosError 的三种常见方法

在使用 Axios 时处理 AxiosError 有几种常见的方法: 使用 try-catch 语句捕获异常: try {const response await axios.get(/api/data);// 处理响应数据 } catch (error) {if (error.response) {// 请求成功但状态码不在 2xx 范围console.log(error.response.data);console.l…

2024比特币减半,Web3的“1995时刻”即将到来

随着比特币减半的到来&#xff0c;加密货币市场迎来了一个关键的转折点。2024年的比特币减半不仅是对比特币供应和挖矿激励的一次重大调整&#xff0c;更是对整个Web3应用领域产生深远影响的事件。 首先&#xff0c;比特币减半的事件本身就为市场带来了一种稀缺性的概念&#…

绝地求生:AUG爆裂弹球黑货箱:街机动漫风格大家会喜欢吗?

大好&#xff0c;我闲游盒&#xff01; 4.10更新后&#xff0c;AUG的新成长型也出来了&#xff0c;更新后我觉得AUG变好用了一点&#xff0c;不知道大家有没有感觉出来&#xff1f; 宝箱概率 本期主角 AUG-爆裂弹球&#xff08;紫色配粉红色&#xff09; 本次的AUG我才升到5级…

蓝桥备赛——组合数、其他技巧

对字符串进行permutations排列组合 from itertools import permutations a abc #对字符串进行permutations排列组合 for i in permutations(a,3):x .join(i)print (x,end ) print (\n------------------------------------) permutations后面的参数&#xff0c;第一个表示…

vue简单使用一(vue的声明)

首先引入vue的js文件&#xff1a; <script src"js/vue.js" type"application/javascript"></script> vue.js文件去vue官网下载即可 html代码 <div class"vuePro"><div> vue的属性信息都得放在这个标签下面哪怕是同级也是…

CSS的属性【all:inherit】有什么奥秘?

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

Scrapy框架内存泄漏问题及解决

说明&#xff1a;仅供学习使用&#xff0c;请勿用于非法用途&#xff0c;若有侵权&#xff0c;请联系博主删除 作者&#xff1a;zhu6201976 一、问题背景及原因 官方文档&#xff1a;Debugging memory leaks — Scrapy 2.11.1 documentation Scrapy是一款功能强大的网络爬虫框…

12个建筑数据分析典型用例

建筑企业面临着众多挑战&#xff0c;包括紧迫的期限、预算限制和复杂的监管要求。 然而&#xff0c;很明显&#xff0c;数据分析可以成为克服这些障碍的重要工具。 建筑行业是数据最密集的市场之一&#xff0c;这就是为什么越来越需要更好的建筑分析和大数据管理。 通过大数据…

【简单讲解下Symfony框架】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

阿里云GPU服务器租用价格一年、1个月和1小时报价明细表

阿里云GPU服务器租用价格表包括包年包月价格、一个小时收费以及学生GPU服务器租用费用&#xff0c;阿里云GPU计算卡包括NVIDIA V100计算卡、T4计算卡、A10计算卡和A100计算卡&#xff0c;GPU云服务器gn6i可享受3折优惠&#xff0c;阿里云服务器网aliyunfuwuqi.com分享阿里云GPU…

面试: 单例模式

目录 一、饿汉单例&#xff08;实现Serializable&#xff09; 1、破坏单例的三种情况 &#xff08;1&#xff09;反射破坏单例 &#xff08;2&#xff09;反序列化破坏单例 &#xff08;3&#xff09;Unsafe破坏单例 2、饿汉单例&#xff08;利用枚举实现&#xff09; 二…

基于STC12C5A60S2系列1T 8051单片机的液晶显示器LCD1602显示自带字库的字符应用

基于STC12C5A60S2系列1T 8051单片机的液晶显示器LCD1602显示自带字库的字符应用 STC12C5A60S2系列1T 8051单片机管脚图STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍LCD1602字符型液晶显示器介绍一、LCD16…

Django框架的基础知识

Django&#xff08;英文发音&#xff1a;dʒŋgəʊ&#xff09;是一个开放源代码的Web应用框架&#xff0c;使用高性能的Python语言编写而成。Django框架的诞生&#xff0c;最初是用来开发和管理Lawrence Publishing Group&#xff08;劳伦斯出版集团&#xff09;旗下的新闻网…

如何开展下拉SEO优化?百度推荐词优化技巧分析

我们在搜索网络信息的时候&#xff0c;经常可以看到搜索框下面会有提示&#xff0c;这就是联想词&#xff0c;也叫下拉词&#xff0c;或者下拉框、推荐词等&#xff0c;指的都是这个地方。在搜索结果中&#xff0c;像百度还会展示“大家还在搜”“大家都在搜”“相关搜索”等推…

​LeetCode解法汇总1026. 节点与其祖先之间的最大差值

目录链接&#xff1a; 力扣编程题-解法汇总_分享记录-CSDN博客 GitHub同步刷题项目&#xff1a; https://github.com/September26/java-algorithms 原题链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给定二叉树的根节点 root&#xff0c;找出…

Nginx的基本使用

目录 介绍Nginx&#xff1a; 其优点有很多&#xff1a; 如何下载Nginx&#xff1a; 下载Nginx 启动Nginx ​编辑 如何用Nginx创建网站 Nginx自带的网站 分析网页 转变ip地址为自己的网页 换内容 换文件 介绍Nginx&#xff1a; Nginx是一个高性能的HTTP和反向代理w…

12-pyspark的RDD算子注意事项总结

目录 相近算子异同总结相近变换算子异同foreach和foreachPartitionfold和reducecoalesce和repatition 相近动作算子异同cache和persist 算子注意事项需要注意的变换算子需要注意的动作算子 PySpark实战笔记系列第三篇 10-用PySpark建立第一个Spark RDD(PySpark实战笔记系列第…

【源码】2024最新最火短剧在线搜索神器源码

搜索神器源码&#xff0c;自带本地数据库500数据&#xff0c;共有6000短剧视频&#xff0c;与短剧猫一样。 搭建环境 PHP7.3 Mysql 5.6 安装教程 1.上传源码到网站目录中 2.修改【admin.php】中&#xff0c; $username ‘后台登录账号’; $password ‘后台登录账号密码’…

leetcode 1766

leetcode 1766 题目 例子 思路 将边的关系&#xff0c;转化为树结构。 记录val 对应的id 列表。 记录是否遍历过节点。 记录id 和对应的深度。 使用dfs&#xff0c; 从根开始遍历。 代码实现 class Solution { private:vector<vector<int>> gcds;//val : the …