GatewayWorker 结合 Laravel 使用的简单案例,重点是在Laravel中使用GatewayClient发送消息

主要流程:GatewayWorker主要负责推送消息给客户端但不接受客户端数据,Laravel主要负责接受客户端数据并处理业务逻辑,然后使用GatewayClient推送数据,

示意图拿一个官方图片哈 http://www.workerman.net/gatewaydoc/work-with-other-frameworks/README.html

跟Laravel结合使用的优点:

1,当 GatewayWorker服务运行时,修改了Laravel业务代码,直接推送业务相关代码到服务器上即可,不用重启GatewayWorker服务;

2,可以尽情使用Laravel提供的各种便利工具处理数据和业务逻辑;

整个项目压缩文件太大不便上传,接下来直接贴代码咯,

本地windows使用步骤:

1,启动GatewayWorker,双击 start_for_win.bat

2,新建一个虚拟域名路径指定到该项目public目录

3,开两个浏览器页面就能互发消息了,如下面两个效果图,

效果图:

项目结构:

ChatServer目录下的4个start_*.php文件是直接复制的官方demo https://github.com/walkor/workerman-chat/tree/master/Applications/Chat

 composer.json 的 require 部分

    "require": {
"php": ">=7.0.0",
"fideloper/proxy": "~3.3",
"laravel/framework": "5.5.*",
"laravel/tinker": "~1.0",
"workerman/gateway-worker" : ">=3.0.0",
"workerman/gatewayclient": "^3.0"
},

start_for_win.bat  本地windows启动脚本 

php app\ChatServer\start_register.php app\ChatServer\start_web.php app\ChatServer\start_gateway.php app\ChatServer\start_businessworker.php
pause

 start.php  服务器上的启动脚本

<?php
/**
* run with command
* php start.php start
*/ ini_set('display_errors', 'on');
use Workerman\Worker; if(strpos(strtolower(PHP_OS), 'win') === 0)
{
exit("start.php not support windows, please use start_for_win.bat\n");
} // 检查扩展
if(!extension_loaded('pcntl'))
{
exit("Please install pcntl extension. See http://doc3.workerman.net/appendices/install-extension.html\n");
} if(!extension_loaded('posix'))
{
exit("Please install posix extension. See http://doc3.workerman.net/appendices/install-extension.html\n");
} // 标记是全局启动
define('GLOBAL_START', 1); require_once __DIR__ . '/vendor/autoload.php'; // 加载所有application/*/start.php,以便启动所有服务
foreach(glob(__DIR__.'/app/ChatServer/start*.php') as $start_file)
{
require_once $start_file;
}
// 运行所有服务
Worker::runAll();

Events.php  参考 http://www.workerman.net/gatewaydoc/work-with-other-frameworks/README.html

<?php

/**
* 用于检测业务代码死循环或者长时间阻塞等问题
* 如果发现业务卡死,可以将下面declare打开(去掉//注释),并执行php start.php reload
* 然后观察一段时间workerman.log看是否有process_timeout异常
*/
//declare(ticks=1); use \GatewayWorker\Lib\Gateway;
class Events
{
// 当有客户端连接时,将client_id返回,让mvc框架判断当前uid并执行绑定
public static function onConnect($client_id)
{
Gateway::sendToClient($client_id, json_encode(array(
'type' => 'init',
'client_id' => $client_id
)));
} // GatewayWorker建议不做任何业务逻辑,onMessage留空即可
public static function onMessage($client_id, $message)
{ }
}
IndexController.php  发送消息用的控制器 参考 http://www.workerman.net/gatewaydoc/work-with-other-frameworks/README.html
<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;
use GatewayClient\Gateway; class IndexController extends Controller
{
public function __construct()
{
Gateway::$registerAddress = '127.0.0.1:1236';
} /**
* web客户端
* @param string $uid
* @param string $to_uid
* @return \Illuminate\Contracts\View\Factory|\Illuminate\View\View
*/
public function client()
{
return view('client');
} /**
* 绑定uid
* @return mixed
*/
public function bind()
{
// 假设用户已经登录,用户uid和群组id在session中
$uid = request('uid');
$client_id = request('client_id'); $res = request()->all();
$res['type'] = 'bind';
$res['time'] = date('H:i:s');
// client_id与uid绑定
Gateway::bindUid($client_id, $uid);
Gateway::sendToUid($uid, json_encode($res));
return response()->json($res);
} /**
* 发送消息
* @return mixed
*/
public function send()
{
$uid = request('uid');
$to_uid = request('to_uid');
$res = request()->all();
$res['type'] = 'send';
$res['time'] = date('H:i:s');
// 向任意uid的网站页面发送数据
Gateway::sendToUid($uid, json_encode($res));
Gateway::sendToUid($to_uid, json_encode($res));
return response()->json($res);
}
}

web.php  路由 

<?php

/*
|--------------------------------------------------------------------------
| Web Routes
|--------------------------------------------------------------------------
|
| Here is where you can register web routes for your application. These
| routes are loaded by the RouteServiceProvider within a group which
| contains the "web" middleware group. Now create something great!
|
*/ Route::get('/', function () {
return view('welcome');
}); Route::match(['post', 'get'], 'index/index/{uid}/{to_uid}', 'IndexController@client'); Route::match(['post', 'get'], 'index/client/{uid}/{to_uid}', 'IndexController@client'); Route::match(['post', 'get'], 'index/bind', 'IndexController@bind'); Route::match(['post', 'get'], 'index/send', 'IndexController@send');

client.blade.php   web客户端视图 

<html>
<head>
<title>{{ request('uid') }} => {{ request('to_uid') }}</title>
<link href="https://cdn.bootcss.com/normalize/8.0.0/normalize.css" rel="stylesheet">
<script src="https://cdn.bootcss.com/jquery/1.12.4/jquery.js"></script>
<script src="https://cdn.bootcss.com/json5/0.5.1/json5.js"></script>
<style>
* {
padding: 0;
margin: 0;
} .chat-main {
width: 600px;
margin: 30px auto;
box-shadow: 0 0 1px gray;
border: 1px solid gray;
line-height: 1.5em;
} .chat-header {
border-bottom: 1px solid gray;
padding: 5px 15px;
} .chat-log {
height: 200px;
overflow-y: auto;
border-bottom: 1px solid gray;
padding: 5px 15px;
} .chat-log dl {
margin: 15px 0;
} .chat-log dl dd {
display: inline-block;
border: 1px solid gray;
padding: 5px 15px;
border-radius: 10px;
border-top-left-radius: 0;
} .chat-log dl.me dd {
border-radius: 10px;
border-top-right-radius: 0;
} .chat-log dl.me {
text-align: right;
} .chat-log dl.me dd {
text-align: left;
} .user-link {
float: right;
} .user-link a {
margin-left: 5px;
} .hide {
display: none;
} .inline-block {
display: inline-block;
} .btn {
text-align: right;
padding: 5px 15px 15px;
} #btn-send {
display: inline-block;
background: white;
border: 1px solid gray;
line-height: 2em;
padding: 0 2em;
outline: none;
} #btn-send:focus {
background: white;
border-color: green;
} #message {
display: block;
width: 570px;
height: 100px;
margin: 15px auto 0;
border: 1px solid gray;
overflow-x: hidden;
overflow-y: auto;
resize: none;
outline: none;
padding: 10px;
} #message:focus {
border-color: green;
} .chat-body > .tpl {
display: none;
}
</style>
</head>
<body> <div class="hide">
bind<input type="text" id="bind" value="{{ url('index/bind') }}"><br>
send<input type="text" id="send" value="{{ url('index/send') }}"><br>
</div> <div class="chat-main">
<div class="chat-header">
<div class="chat-title inline-block">
{{ request('uid') }} => {{ request('to_uid') }}
</div>
<div class="user-link inline-block">
<span class="inline-block">模拟用户</span>
<a class="inline-block" href="{{ url('index/index',['uid'=>1111,'to_uid'=>2222]) }}">1111</a>
<a class="inline-block" href="{{ url('index/index',['uid'=>2222,'to_uid'=>1111]) }}" target="_blank">2222</a>
</div>
</div>
<div class="chat-body">
<div class="chat-log"> </div>
<dl class="tpl">
<dt>1111(12:00:00)</dt>
<dd>aaaabbbbbb</dd>
</dl>
</div>
<div class="chat-footer">
<form action="" id="form">
<div class="hide">
cliend_id<input type="text" name="client_id" id="client_id" value="{{ request('uid') }}"><br>
uid<input type="text" name="uid" id="uid" value="{{ request('uid') }}"><br>
to_uid<input type="text" name="to_uid" value="{{ request('to_uid') }}"><br>
</div>
<textarea name="message" id="message" cols="30" rows="10"></textarea>
<div class="btn">
<button type="button" id="btn-send">发 送</button>
</div>
</form>
</div>
</div> <script> /**
* 与GatewayWorker建立websocket连接,域名和端口改为你实际的域名端口,
* 其中端口为Gateway端口,即start_gateway.php指定的端口。
* start_gateway.php 中需要指定websocket协议,像这样
* $gateway = new Gateway(websocket://0.0.0.0:7272);
*/
ws = new WebSocket("ws://127.0.0.1:7272");
// 服务端主动推送消息时会触发这里的onmessage
ws.onmessage = function (e) {
// json数据转换成js对象
var data = JSON5.parse(e.data);
var type = data.type || '';
switch (type) {
// Events.php中返回的init类型的消息,将client_id发给后台进行uid绑定
case 'init':
$('#client_id').val(data.client_id);
// 利用jquery发起ajax请求,将client_id发给后端进行uid绑定
$.get($('#bind').val(), $('#form').serialize(), function (res) {
console.log('bind', res);
}, 'json');
break;
case 'send':
var $tpl = $('.chat-body>.tpl').clone().show(); if (data.uid == $('#uid').val()) {
$tpl.find('dt').html('(' + data.time + ') ' + data.uid);
$tpl.addClass('me')
} else {
$tpl.find('dt').html(data.uid + ' (' + data.time + ')');
} $tpl.find('dd').html(data.message.replace(/\n/gim, '<br>')); $('.chat-log').append($tpl); scrollBottom();
break;
// 当mvc框架调用GatewayClient发消息时直接alert出来
default:
console.log('default', e.data);
}
}; $('#form').submit(function (e) {
return false;
}); var isScrollBottom = true; function scrollBottom(){
if (isScrollBottom) {
$('.chat-log').scrollTop($('.chat-log')[0].scrollHeight);
}
} $('#btn-send').click(function (e) {
if ($.trim($('#message').val())) {
$.get($('#send').val(), $('#form').serialize(), function (res) {
console.log('send', res);
$('#message').val('');
scrollBottom();
}, 'json');
}
}); $('.chat-log').scroll(function (e) {
var outerHeight = $(this).outerHeight();
var scrollTop = $(this).scrollTop();
var scrollHeight = $(this)[0].scrollHeight;
if (outerHeight + scrollTop >= scrollHeight - 15) {
isScrollBottom = true;
} else {
isScrollBottom = false;
}
})
</script> </body>
</html>

==============================================

本文链接 https://www.cnblogs.com/stumpx/p/9156850.html

==============================================

GatewayWorker+Laravel demo的更多相关文章

  1. 使用GatewayWorker 开发个即时聊天demo

    前言: 上手册以示尊重:https://www.kancloud.cn/walkor/gateway-worker/326138: https://www.cnblogs.com/fuqiang88/ ...

  2. 聊天室(下篇)GatewayWorker 与 Laravel 的整合

    思路 上一篇大概梳理了一下 GatewayWorker 的基础知识.这篇就来准备整合 GatewayWorker 到 Laravel. GatewayWorker 是基于 Socket 监听的服务器框 ...

  3. 02 workerman之GatewayWorker简单的demo 实现两端发送消息

    前端代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <t ...

  4. laravel的中间件demo

    过滤器已经被废除...刚学才两天,蛋疼 创建一个中间件 ./artisan make:middleware TestMiddleware 大概代码 <?php namespace App\Htt ...

  5. Laravel 5.3 请求处理管道详解

    对于一个Web应用来说,在一个请求真正处理前,我们可能会对请求做各种各样的判断,然后才允许后续处理. 我们通常的做法: Script 01.php Script 02.php 优点:直观,容易理解 缺 ...

  6. 在window下配置laravel开发环境

    1.由于有一点php基础,所以非常想更进一步,就选择据说在国外最流行的php框架来学习了,laravel框架,官网上介绍是为艺术而生,从知乎和一些论坛上看到,laravel学起来并不简单,首先配置问题 ...

  7. laravel 指定 版本安装

    composer create-project laravel/laravel=5.0.* --prefer-dist composer create-project laravel/laravel= ...

  8. php后台开发(二)Laravel框架

    php后台开发(二)Laravel框架 为了提高后台的开发效率,往往需要选择一套适合自己的开发框架,因此,选择了功能比较完善的Laravel框架,仔细学来,感觉和Python语言的框架Django非常 ...

  9. Laravel 5 基础(十二)- 认证

    Laravel 出厂已经带有了用户认证系统,我们来看一下 routes.php,如果删除了,添加上: Route::controllers([ 'auth' => 'Auth\AuthContr ...

随机推荐

  1. 再理解HDFS的存储机制

    再理解HDFS的存储机制 1. HDFS开创性地设计出一套文件存储方式.即对文件切割后分别存放: 2. HDFS将要存储的大文件进行切割,切割后存放在既定的存储块(Block)中,并通过预先设定的优化 ...

  2. Linux 文本编辑

    文本编辑: 查看文本内容:    cat:将文件连接并显示    -n:显示时将文件每一行编号        tac:类似于cat,但其功能是逆序显示每一行文件       linlin@ubuntu ...

  3. MJRefresh实现刷新(使用它的Block方法)

    MJRefresh实现刷新(使用它的Block方法)   // // YFMVCPostListViewController.m // iOS122 // // Created by 颜风 on 15 ...

  4. 配置server禁止全部非法域名 訪问自己的server

    1.Apache2.4.1曾经: 第一种 直接拒绝訪问 打开 httpd.conf  文件,将一下配置追加到文件最后. #直接拒绝全部非法域名 <VirtualHost *:80> Ser ...

  5. SVN代码丢失惊魂

    吓死了吓死了!要是那些代码丢了的话,要重新码一遍,我宁愿去吃屎. 某天快下班走人的时候,从SVN服务器update了本地代码,结果发现代码变回了上个月的样子.看SVN的日志,发现提交记录从6月22日一 ...

  6. C项目实践--俄罗斯方块(2)

    在VS中新建win32 Application Proj,选择Empty ,完成TetrisWin项目创建.新建tetris.c和tetris.h两个文件,打开tetris.h文件. 首先要包括的是可 ...

  7. 大括号对struct进行初始化

    1 partial initialization 即所谓的部分初始化. 这个时候,无论该struct变量是static的还是automic的,未显式初始化的成员都会被初始化为默认值.

  8. 授权QQ登录的qq端前端页面变迁

    ac_type = 'qq' if ac_type == 'qq': myid, mypwd = qq_key xp = '/html/body/div/div/div[2]/div/div/div/ ...

  9. ABAP WEBRFC

    通过WEBRFC实现在网页下载SMW0上传的文件 FUNCTION zhr_download_test. *"---------------------------------------- ...

  10. XMU 1612 刘备闯三国之桃园结义 【二分】

    1612: 刘备闯三国之桃园结义 Time Limit: 1000 MS  Memory Limit: 128 MBSubmit: 181  Solved: 12[Submit][Status][We ...