• Workerman 聊天系统


    文档、文件下载地址

    聊天库: Workerman 
    文档: http://doc2.workerman.net/
    手册: http://www.workerman.net/gatewaydoc/
    下载Demo: http://doc2.workerman.net/
     CatewayWorker 服务端安装 Liunx: http://www.workerman.net/download/GatewayWorker.zip
     CatewayWorker 服务端安装 Window: http://www.workerman.net/download/GatewayWorker-for-win.zip

    GatewayWorker框架与MVC结合总体原则:

    现有mvc框架项目与GatewayWorker独立部署互不干扰

    所有的业务逻辑都由网站页面post/get到mvc框架中完成

    GatewayWorker不接受客户端发来的数据,即GatewayWorker不处理任何业务逻辑,GatewayWorker仅仅当做一个单向的推送通道

    仅当mvc框架需要向浏览器主动推送数据时才在mvc框架中调用Gateway的API GatewayClient完成推送。

    MVC框架(客户)端安装

    composer 安装 GatewayClinet  客户端

    composer require workerman/gatewayclient
    //使用时引入vendor/autoload.php 类似如下: use GatewayClient\Gateway; require_once '真实路径/vendor/autoload.php';

     或是下载源文件到任意目录,手动引入 GatewayClient/Gateway.php, 类似如下:

    use GatewayClient\Gateway;
    require_once '真实路径/GatewayClient/Gateway.php';

    服务端安装

    CatewayWorker 服务端安装,可以直接下载源文件

    composer require workerman/gateway-worker-for-win //安装Windows版本的gateway
    composer require workerman/gateway-worker //安装Linux版本的gateway

    服务端项目目录结构,GatewayWorker单独部署的

    ├── Applications // 这里是所有开发者应用项目
    │   └── YourApp  // 其中一个项目目录,目录名可以自定义
    │       ├── Events.php // 开发者只需要关注这个文件
    │       ├── start_gateway.php // gateway进程启动脚本,包括端口号等设置
    │       ├── start_businessworker.php // businessWorker进程启动脚本
    │       └── start_register.php // 注册服务启动脚本
    │
    ├── start.php // 全局启动脚本,此脚本会依次加载Applications/项目/start_*.php启动脚本
    │
    └── vendor    // GatewayWorker框架和Workerman框架源码目录,此目录开发者不用关心

    服务端文件操作

    创建 start.bat,一般放根目录,Window 启动文件,双击

    php worker\win\start_register.php worker\win\start_gateway.php worker\win\start_businessworker.php
    pause

    创建 start.php,一般放根目录,Liunx 启动文件,命令行  php start.php start

    /**
     * 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__ . '/worker/vendor/autoload.php';
    
    // 加载所有Applications/*/start.php,以便启动所有服务, 此处我放根目录/worker/win/ 下,其他要启动的文件是start.bat文件中指定的文件
    foreach(glob(__DIR__.'/worker/win/start*.php') as $start_file)
    {
        require_once $start_file;
    }
    // 运行所有服务
    Worker::runAll();

    两种启动方式 ,start_gateway.php start_businessworker.php start_register.php分别是进程启动脚本,三个脚本统一由根目录的start.php/ start.bat启动。

    文件说明

    start_register.php 为注册服务启动脚本,用于协调GatewayWorker集群内部Gateway与Worker的通信,参见Register类使用一节。

    注意:客户端不要连接Register服务端口,客户端应该连接Gateway端口

    use \Workerman\Worker;
    use \GatewayWorker\Register;
    
    // 自动加载类,根据需求自定义
    require_once __DIR__ . '/../../vendor/autoload.php';
    
    // register 必须是text协议,服务注册地址
    $register = new Register('text://127.0.0.1:1234');
    
    // 如果不是在根目录启动,则运行runAll方法
    if(!defined('GLOBAL_START'))
    {
        Worker::runAll();
    }

    start_gateway.php   为gateway进程启动脚本,主要定义了客户端连接的端口号、协议等信息,具体参见 Gateway类的使用一节。

    use \Workerman\Worker;
    use \Workerman\WebServer;
    use \GatewayWorker\Gateway;
    use \GatewayWorker\BusinessWorker;
    use \Workerman\Autoloader;
    
    // 自动加载类
    require_once __DIR__ . '/../vendor/autoload.php';
    /**
     * 与GatewayWorker建立websocket连接,域名和端口改为你实际的域名端口,
     * 其中端口为Gateway端口,即start_gateway.php指定的端口。
     * start_gateway.php 中需要指定websocket协议,像这样
     * $gateway = new Gateway(websocket://0.0.0.0:7272);
     * js部分:
     * ws = new WebSocket("ws://your_domain.com:7272");
     */
    // gateway 进程,这里使用Text协议,可以用telnet测试
    
    $context = array(
        // 更多ssl选项请参考手册 http://php.net/manual/zh/context.ssl.php
        'ssl' => array(
            // 请使用绝对路径
            'local_cert'                 => __dir__.'/certificate/server.pem', // 也可以是crt文件
            'local_pk'                   => __dir__.'/certificate/server.key',
            'verify_peer'               => false,
            // 'allow_self_signed' => true, //如果是自签名证书需要开启此选项
        )
    );
    // websocket协议(端口任意,只要没有被其它程序占用就行)
    $gateway = new Gateway("websocket://0.0.0.0:8899", $context);
    // 开启SSL,websocket+SSL 即wss
    $gateway->transport = 'ssl';
    // gateway名称,status方便查看
    $gateway->name = 'YourAppGateway';
    // gateway进程数
    $gateway->count = 4;
    // 本机IP,分布式部署时使用内网IP,GatewayWorker与GatewayClient不在同一台服务器使用当前服务器内网IP,如果不在同一个内网改为公网IP
    $gateway->lanIp = '127.0.0.1';
    // 内部通讯起始端口,假如$gateway->count=4,起始端口为4000
    // 则一般会使用4000 4001 4002 4003 4个端口作为内部通讯端口 
    $gateway->startPort = 2900;
    // 服务注册地址
    $gateway->registerAddress = '127.0.0.1:1234';
    
    // 心跳间隔
    //$gateway->pingInterval = 10;
    // 心跳数据
    //$gateway->pingData = '{"type":"ping"}';
    
    /* 
    // 当客户端连接上来时,设置连接的onWebSocketConnect,即在websocket握手时的回调
    $gateway->onConnect = function($connection)
    {
        $connection->onWebSocketConnect = function($connection , $http_header)
        {
            // 可以在这里判断连接来源是否合法,不合法就关掉连接
            // $_SERVER['HTTP_ORIGIN']标识来自哪个站点的页面发起的websocket链接
            if($_SERVER['HTTP_ORIGIN'] != 'http://kedou.workerman.net')
            {
                $connection->close();
            }
            // onWebSocketConnect 里面$_GET $_SERVER是可用的
            // var_dump($_GET, $_SERVER);
        };
    }; 
    */
    
    // 如果不是在根目录启动,则运行runAll方法
    if(!defined('GLOBAL_START'))
    {
        Worker::runAll();
    }

    start_businessworker.php 为businessWorker进程启动脚本,也即是调用Events.php的业务处理进程,具体参见 BusinessWorker类的使用一节。

    use \Workerman\Worker;
    use \Workerman\WebServer;
    use \GatewayWorker\Gateway;
    use \GatewayWorker\BusinessWorker;
    use \Workerman\Autoloader;
    
    // 自动加载类
    require_once __DIR__ . '/../../vendor/autoload.php';
    
    // bussinessWorker 进程
    $worker = new BusinessWorker();
    // worker名称
    $worker->name = 'YourAppBusinessWorker';
    // bussinessWorker进程数量
    $worker->count = 4;
    // 服务注册地址
    $worker->registerAddress = '127.0.0.1:1234';
    
    // 如果不是在根目录启动,则运行runAll方法
    if(!defined('GLOBAL_START'))
    {
        Worker::runAll();
    }

    注意:注册端口 

    文件: start_register.php     start_gateway.php     start_businessworker.php 中的注册地址要一样。

    //start_gateway.php
    // 服务注册地址
    $gateway->registerAddress = '127.0.0.1:1238';
    
    //start_businessworker.php
    // 服务注册地址
    $worker->registerAddress = '127.0.0.1:1238';
    
    //start_register.php
    // register 必须是text协议
    $register = new Register('text://0.0.0.0:1238');

    Events.php 所有业务代码都在这里

     /* 
    * 用于检测业务代码死循环或者长时间阻塞等问题 * 如果发现业务卡死,可以将下面declare打开(去掉//注释),并执行php start.php reload * 然后观察一段时间workerman.log看是否有process_timeout异常 */
    //declare(ticks=1); use \GatewayWorker\Lib\Gateway; /** * 主逻辑 * 主要是处理 onConnect onMessage onClose 三个方法 * onConnect 和 onClose 如果不需要可以不用实现并删除 */ class Events { /** * 当客户端连接时触发 * 如果业务不需此回调可以删除onConnect * @param int $client_id 连接id * @desc $client_id 为系统自动生成 */ public static function onConnect($client_id) { // 向当前client_id发送数据 /*Gateway::sendToClient($client_id, "Hello $client_id\r\n"); // 向所有人发送 Gateway::sendToAll("$client_id login\r\n");*/ } /** * 当客户端发来消息时触发 * @param int $client_id 连接id * @param mixed $message 具体消息 */ public static function onMessage($client_id, $message) { // 向所有人发送 Gateway::sendToAll("$client_id said $message\r\n"); } /** * 当用户断开连接时触发 * @param int $client_id 连接id */ public static function onClose($client_id) { } }

    启动与停止

    注意:Workerman启动停止等命令都是在命令行中完成的。

    启动

    以debug(调试)方式启动

    php start.php start

    以daemon(守护进程)方式启动

    php start.php start -d

    停止

    php start.php stop

    重启

    php start.php restart

    平滑重启

    php start.php reload

    查看状态

    php start.php status

    debug和daemon方式区别

    1、以debug方式启动,代码中echo、var_dump、print等打印函数会直接输出在终端。

    2、以daemon方式启动,代码中echo、var_dump、print等打印会默认重定向到/dev/null文件,可以通过设置Worker::$stdoutFile = '/your/path/file';来设置这个文件路径。

    3、以debug方式启动,终端关闭后workerman会随之关闭并退出。

    4、以daemon方式启动,终端关闭后workerman继续后台正常运行。

  • 相关阅读:
    计算机编码总结
    将TOMCAT设置成为NT服务
    java 操作oracle 序号器相关
    java事件机制
    c# Semaphore(信号量)
    C#中异步和多线程的区别
    c#多线程调用有参数的方法
    解决TCP网络传输“粘包”问题
    高性能socket设计实现
    c# Buffer学习笔记
  • 原文地址:https://www.cnblogs.com/xuey/p/15817018.html
Copyright © 2020-2023  润新知