美文网首页
easyswoole websocket 之解决负载均衡后无法

easyswoole websocket 之解决负载均衡后无法

作者: 小李不是你 | 来源:发表于2019-12-24 10:01 被阅读0次

一、需求

1、问题

由于需要做负载均衡,但做了负载均衡之后,当请求向新的服务器分发时,websocketfd则从 0开始,故当pcapp不在同一服务器时,则会出现pc无法直接向app传递消息。

具体问题:PC发送请求,被转发到服务器AAPP发送请求,被转发到服务器B,这时PCAPP由于不在同一服务器,故不能直接通过websocket进行传递消息。

二、环境配置

1、准备

使用三台服务器做负载均衡,分别为 ABC,其中A用来做负载均衡,BC存储websocket源码

2、负载均衡配置
  • 主服务器 A配置
  upstream taishan {
      server server B:9502  weight=1;
      server server C:9502  weight=1;
  }


  server {
    listen 80;
    server_name domain;

    location ~* /wss(.*)$ {
         proxy_redirect off;
         proxy_pass http://taishan;
         proxy_set_header Host $host;
         proxy_set_header X-Real-IP $remote_addr;
         proxy_set_header X-Forwarded-For $remote_addr:$remote_port;
         proxy_http_version 1.1;
         proxy_set_header Upgrade $http_upgrade;
         proxy_set_header Connection "upgrade";
         proxy_set_header DIY-RUN-ENV "TEST";
    }

    location / {
        #proxy_redirect  off;
        #proxy_set_header X-Real-IP $remote_addr;
        #proxy_set_header X-Forwarded-For $remote_addr:$remote_port;
        #proxy_set_header Host $http_host;
        #proxy_http_version 1.1;

        proxy_pass http://taishan;

    }

 }

ps:若不想把服务器B、C的9502端口暴露在外,则可以用反向代理,如果是阿里云服务器,则需要在阿里云开放端口

3、主服务器redis配置

参考链接:https://blog.joniding.com/index.php/archives/26/

4、php redis扩展、swoole扩展

具体安装请使用搜索引擎

三、实现思路

go(function () use ($conf){

    $redis = new \EasySwoole\Redis\Redis(new RedisConfig([
        'host'      => $conf['host'],
        'port'      => $conf['port'],
        'auth'      => $conf['auth'],
    ]));

    $ip = UtilHelper::get_server_ip();

    echo "订阅频道为:". $this->channel.'_'.$ip . "\n";

    //订阅完成设置已订阅
    $redis->set($this->channel.'_'.$ip,1);

    //订阅
    $redis->subscribe(function ($redis, $pattern, $str) {
        echo '订阅频道已收到消息' . "\n";
        $data = json_decode($str,true);
        echo $str . "\n";

        $fd_list = isset($data['fd_list'])?$data['fd_list']:'';
        $server_ip = UtilHelper::get_server_ip();

        //判断服务器ip与绑定设备的ip是否一致
        if ($server_ip == $fd_list['server_ip']){
            echo 'ip is ok' . "\n";
            $push_arr = [
                'receive_data'  => $data['receive_data'],
                'fd_list'       => $data['fd_list'],
                'current_fd'   => $data['current_fd'],
                'type'         => $data['type'],
                'msg'          => $data['msg'],
                'status'       => isset($data['status'])?$data['status']:100200,
                'flag'         => $data['flag']
            ];
            TaskManager::getInstance()->async(new BroadCastTask($push_arr));
        }

    }, $this->channel.'_'.$ip);
});

如上述代码所示,在配置文件中设置好服务器ip,然后使用redis订阅当前服务器频道,以服务器ip来区分,如:redis_192.168.1.10redis_192.168.1.12,接下来就是请求时,判断PCAPP是否在同一服务器,如果不在则发消息给订阅频道,代码示例如下:

   /**
     * 发布消息
     * @param $channel
     * @param $message
     * @return bool
     * @author:joniding
     * @date:2019/12/20 10:47
     */
    public function lPublish($channel,$message)
    {
        $conf = Config::getInstance()->getConf('REDIS');


        go(function () use ($conf,$channel,$message){
            $redis = new \EasySwoole\Redis\Redis(new RedisConfig([
                'host'      => $conf['host'],
                'port'      => $conf['port'],
                'auth'      => $conf['auth'],
            ]));

           $redis->publish($channel,$message);
        });
        return true;
    }
    
    //调用示例
    $subcribe = new Subscribe();
    $result   = $subcribe->lPublish(self::$channel.'_'.$fd_server_ip,json_encode($push_arr));

注意:
* 使用redis订阅时,建议使用协程或注册进程,若不使用协程,则订阅的回调不会触发
* 若出现订阅回调出现多次收到回调消息,则是因为订阅了多次该频道,解决方案,第一次订阅时,在redis设置一个已订阅的标识字段

相关文章

  • easyswoole websocket 之解决负载均衡后无法

    一、需求 1、问题 由于需要做负载均衡,但做了负载均衡之后,当请求向新的服务器分发时,websocket的fd则从...

  • Apache Flume基础(四)负载均衡与容错机制

    Apache Flume基础(四)负载均衡与容错机制 负载均衡 负载均衡是用于解决一台机器(一个进程)无法解决所有...

  • WebSocket负载均衡

    这一期我们来说一下WebSocket负载均衡的问题 我们知道负载均衡在IT中是个很重要的概念,尤其是在互联网行业...

  • Nginx (4)

    Nginx之负载均衡 Nginx 通过Upstream 模块进行负载均衡。 upstream 支持的负载均衡算法N...

  • Nginx-负载均衡

    章节目录 什么是负载均衡 使用负载均衡要解决的问题 基于LVS的中间件架构GSLBSLB四层负载均衡和七层负载均衡...

  • SSO单点登录处理

    集群环境下的登录 在集群环境下可能会出现要求用户多次登录的情况(配置了负载均衡后无法实现Session共享) 解决...

  • 负载均衡-interview

    简介 Load Balance负载均衡是用于解决一台机器(一个进程)无法解决所有请求而产生的一种算法。 服务器负载...

  • Java websocket 在负载均衡上实现单对单通讯

    由于公司项目需要用到websocket进行通讯,服务器采用负载均衡模式。由于WebSocketSession对象...

  • 负载均衡

    负载均衡要解决什么问题 负载均衡主要解决以下几个问题: 解决并发压力,提高应用处理性能(增加吞吐量,加强网络处理能...

  • 4层负载均衡服务偶发1S请求的故障分析

    4层负载均衡平台上线后,在公司推广比较顺利,有很多业务使用。我们支持两种服务:静态负载均衡与动态负载均衡,静态负载...

网友评论

      本文标题:easyswoole websocket 之解决负载均衡后无法

      本文链接:https://www.haomeiwen.com/subject/osmonctx.html