依赖 laravel(predis)、 redis、 nodejs(ioredis,socket.io)
1、修改config\app.php
providers数组 添加 ‘Illuminate\Broadcasting\BroadcastServiceProvider’,
2、修改广播驱动方式为 config\broadcasting.php
‘default’ => env(‘BROADCAST_DRIVER’, ‘redis’), 改为redis驱动
使用redis作为php和js的通信方式。
3、配置config\database.php
配置redis服务连接参数
定义一个被广播的事件
<?php
namespace App\Events;
use App\Events\Event;
use Illuminate\Queue\SerializesModels;
use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
use Illuminate\Support\Facades\Session;
class MessageBroadcastEvent extends Event implements ShouldBroadcast
{
use SerializesModels;
public $users;
public $message = array();
protected $channel;
/**
* Create a new event instance.
*
* @return void
*/
public function __construct($users, $message, $channel)
{
$this->users = $users;
$this->message = array(
'id' => $message['id'],
'title' => $message['title'],
'content' => $message['content'],
'url' => $message['url'],
'time' => date('m-d H:i:s', strtotime($message['created_at']))
);
$this->channel = $channel;
}
/**
* Get the channels the event should be broadcast on.
* 广播到哪个频道
* @return array
*/
public function broadcastOn()
{
return [$this->channel];
}
}
默认情况下,Event中的所有public属性都会被序列化后广播。上面的例子中就是users,message 两个属性。也可以
使用broadcastWith这个方法,明确的指出要广播什么数据。例如:
public function broadcastWith(){
return ['message ' => $this->message ];
}
Redis和Websocket服务器
依赖的就是redis的sub/pub功能
启动一个node websocket服务器来和client通信,我们使用socket.io
node 服务端代码 保存为 index.js 放在node服务目录
var app = require('http').createServer(handler);
var io = require('socket.io')(app);
var Redis = require('ioredis');
var redis = new Redis('6379', '192.168.10.10');
//连接redis服务器
//监听客户端端口 这里是 6001
app.listen(6001, function() {
console.log('Server is running!');
});
function handler(req, res) {
res.writeHead(200);
res.end('');
}
io.on('connection', function(socket) {
console.log('connected');
});
redis.psubscribe('*', function(err, count) {
console.log(count);
});
redis.on('pmessage', function(subscribed, channel, message) {
console.log(subscribed);
console.log(channel);
console.log(message);
//发送到客户端的数据
message = JSON.parse(message);
io.emit(channel + ':' + message.event, message.data);
});
客户端代码,只要客户端需要被广播的页面正确引用 node socket.io 模块客户端js文件(自行将这个客户端模块文件放到项目public目录)
<script src="js/socket.io/node_modules/socket.io-client/socket.io.js"></script>
//客户端也使用socket.io,测试代码:控制台打印输出
//连接socket服务器
var socket = io('http://localhost:6001');
socket.on('connection', function (data) {
console.log(data);
});
//收听的频道
socket.on('channel-{{ Session::get('shop')->id }}:App\\Events\\MessageBroadcastEvent', function(data) {
//控制台输出广播消息
console.log(message);
//这里可以根据收到的消息,做一些改变页面结构的工作……
});
//可以收听多个频道
socket.on('channel-system:App\\Events\\MessageBroadcastEvent', function(data){
console.log(data);
//这里可以根据收到的消息,做一些改变页面结构的工作……
});
//控制台输出连接信息
console.log(socket);
项目中触发事件
在控制器或者在路由匿名函数中都可以直接调用广播事件
1、控制器中直接调用
//发送给哪些用户 id 。 这里定义消息接收用户,是在前台用于检测登陆用户是否在这个数组中,存在则做出相应的即时提醒。
//注意:其实广播消息都会被发送到对应的频道的。
$users = array(1, 2);
//这里可以保存发送消息到 messages 表
$message = new Message();
$message->title = '您的店铺有一条新销售单';
$message->content = '您的店铺有一条新销售单,单号1000000';
$message->message_type_id = 1;
$message->status = 0;
$message->url = 'http://www.xxx.com';
$message->save();
//保存发送用户 到 user_message 表
$userMessage = array();
$time = date("Y-m-d H:i:s");
foreach ($users as $user) {
$tmp = array(
'created_at' =>$time,
'updated_at' =>$time,
'user_id' =>$user,
'message_id' =>$message->id,
'read' =>0
);
$userMessage[] = $tmp;
}
UserMessage::insert($userMessage);
//广播的频道
//我们以店铺id来标识频道,这样前端用户页面也根据店铺id标识来收听自己店铺频道,就能做到店铺广播消息消息只能广播到本店铺用户
$channel = 'channel-' . Session::get('shop')->id;
//$channel = 'channel-system'; //其他频道
//$response = event(new MessageBroadcastEvent($users, $message, $channel));
Event::fire(new MessageBroadcastEvent($users, $message, $channel)); //这两种方式都可以触发事件
2、路由中直接调用
//示例代码
Route::get('/event', function(){
Event::fire(new \App\Events\SomeEvent(3));
return "hello world";
});
使用:
必须开启 node websocket 服务端。我在本机windows下C盘安装node,服务端代码就放在这个目录下,进入cmd
终端,执行命令,node index.js 启动服务端。
打开包含有 socket.io 代码的客户端页面,等待被广播
触发我们的后台广播事件(执行相应的控制器代码)