easyswoole websocket 之解决负载均衡后无法通讯问题

一、需求

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设置一个已订阅的标识字段

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容