基于SSE的全平台可用WebPush方案

613 阅读4分钟

因为众所周知的原因,chrome的浏览器推送始终不能用。本文推荐一种使用SSE技术替代厂商通道实现全平台浏览器推送的方案。

原理

浏览器推送

浏览器推送.png

SSE推送

SSE推送.png

增加成本

  1. redis,服务端需要单例或集群部署的redis实例(可与其他服务共用);
  2. sse-broker,一个或多个sse-broker实例,每个实例占用10M内存,每个连接可按2k内存计算;

步骤

安装运行sse-broker

  1. 下载sse-broker:Releases · sssxyd/go-sse-broker
  2. 安装,windows下解压,linux下 rpm -Uvh xxx.rpm
  3. 编辑config.toml windows下在解压所在目录,linux下在 /etc/sse-broker/config.toml
  4. 运行服务,windows下双击sse-broker.exe, linux下 systemctl start sse-broker

config.toml需要编辑以下配置项

[server]
port = 8080

[jwt]
secret = "please_modify"

[redis]
addrs = ["please_modify_1:6379", "please_modify_2:6379"]
password = "please_modify"
db = 0

配置Nginx/Openresty

vim your_website.conf

        # 暴露 sse-broker 的api,生产环境不建议 
        location ~ ^/(info|send|kick|token)$ {
                proxy_redirect off;
                proxy_set_header        Host $host;
                proxy_set_header        X-Real-IP $remote_addr;
                proxy_set_header        X-Forwarded-For $proxy_add_x_forwarded_for;

                proxy_pass http://your_sse_work_ip:8080$request_uri;
        }

        # sse-broker 长连接地址
        location /events {
                proxy_redirect off;
                proxy_set_header        Host $host;
                proxy_set_header        X-Real-IP $remote_addr;
                proxy_set_header        X-Forwarded-For $proxy_add_x_forwarded_for;

                # 关闭缓冲
                proxy_buffering off;
                proxy_cache off;

                # 超时配置,确保超时时间大于心跳间隔时间
                proxy_read_timeout 600s;

                proxy_pass http://your_sse_work_ip:8080/events;
        }

运行Demo页面

  1. 复制附录中的 sse-demo.html 到你的web站点根目录下
  2. 浏览器访问:your.domain.com/sse-demo.ht…
  3. 输入uid并连接sse服务
  4. 向另一个uid发送通知或消息

image.png

对比

本方案以当前站点的私有SSE通道替代了浏览器厂商的推送通道,优缺点如下:

缺点

  • 当前站点关闭后,sse通道也关闭,无法推送;而浏览器推送方案,只要浏览器实例不关闭,推送仍然可用
  • 需要redis,因为sse-broker依赖redis实现集群和LastEventId机制

优点

  • 全平台可用,包括手机端;支持http的客户端均可用;
  • 与业务紧密结合,sse-broker支持单用户多设备同时登陆,业务系统只需要通过用户ID进行推送,可通过其回调管理用户/设备在线状态

注意

  • 获取token及发送通知的接口,不应暴露在外网,应在业务系统中实现对应接口,鉴权成功后再调用sse-broker的接口
  • sse长连接在配置时注意关闭缓冲、配置远大于心跳(默认30秒)的超时时间,以确保sse连接的心跳能实时发送

附录

sse-demo.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>SSE Demo</title>
    <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
    <style>
        #messages {
            width: 100%;
            height: 300px;
            overflow-y: auto;
            border: 1px solid #ccc;
            margin-top: 20px;
            padding: 10px;
            background-color: #f9f9f9;
        }
        .message {
            margin-bottom: 10px;
        }
        input {
            min-width: 400px;
        }
        textarea {
            min-width: 400px;
            min-height: 50px;
        }
    </style>
    <script type="text/javascript">
    // 请求通知权限
    function requestNotificationPermission() {
        if ("Notification" in window) {
            // 检查是否已经获取权限
            if (Notification.permission === "granted") {
                // 权限已获取,直接继续
                console.log("通知权限已授予");
            } else if (Notification.permission !== "denied") {
                // 如果用户尚未授予或拒绝权限,则请求权限
                Notification.requestPermission().then(permission => {
                    if (permission === "granted") {
                        console.log("通知权限已授予");
                    } else {
                        console.log("用户拒绝通知权限");
                    }
                }).catch(error => {
                    console.log("通知权限请求出错", error);
                });
            }
        } else {
            console.log("浏览器不支持通知");
        }
    }


    function generateUUID() {
        return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {
            var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
            return v.toString(16);
        });
    }

    function getDevice() {
        // 获取设备标识,如果没有则生成一个
        // device标识用于区分不同设备,同一个device同一时间只能有一个连接,后登陆的会将前一个挤下线
        // sse的lastEventId机制是基于devcie而不是uid的
        let device = localStorage.getItem('device');
        if (!device) {
            device = generateUUID();
            localStorage.setItem('device', device);
        }
        return device;
    }

    function getToken(uid, device) {
        // 获取一个绑定uid和device的3秒内有效的 sse token
        return new Promise((resolve, reject) => {
            $.ajax({
                url: '/token',
                method: 'POST',
                contentType: 'application/json',
                data: JSON.stringify({
                    uid: uid,
                    device: device,
                    ttl: 3  // 3秒有效期
                }),
                success: function(response) {
                    if(response.code != 1) {
                        reject('[' + response.code + "] " + response.msg);
                    }
                    else {
                        resolve(response.result);
                    }
                },
                error: function() {
                    reject('get token failed');
                }
            });
        });
    }

    // 向消息区域添加消息并滚动到底部
    function appendMessage(type, title, data) {
        const messageDiv = $('<div class="message"></div>');
        messageDiv.html('<strong>' + title + ':</strong> ' + data);
        $('#messages').append(messageDiv);

        // 自动滚动到底部
        $('#messages').scrollTop($('#messages')[0].scrollHeight);
    }

    // 显示通知
    function showNotification(message) {
        if (Notification.permission === "granted") {
            new Notification("Notice", {
                body: message,
                icon: '' 
            });
        }
        else {
            appendMessage('event', 'Notice', message);
            appendMessage('warning', 'Warning', '通知失败,请允许本站点的通知权限');
        }
    }


    function startSSE() {
        requestNotificationPermission();
        const uid = $('#uid').val();
        if (uid == '') {
            alert('please input uid');
            return;
        }
        const device = getDevice();
        // 每次重连都重新获取token;
        // 此处可改为向业务系统请求token,鉴权后由业务系统调用sse的token接口生成token
        getToken(uid, device).then(token => {
            return connectSSE(token, device);
        }).catch(error => {
            setTimeout(function() {
                startSSE();  // 重连
            }, 5000);  // 5 秒后重连
        });
    }

    function connectSSE(token, device) {
        return new Promise((resolve, reject) => {
            const eventSource = new EventSource('/events?token=' + token + '&device=' + device);

            eventSource.onopen = function() {
                appendMessage('info', 'Info', '连接成功');
                $('#yourId').text('你的ID: ' + $('#uid').val());
                $('#user-form').hide();
                $('#send-form').show();
            };

            eventSource.onmessage = function(event) {
                appendMessage('message', 'Message', event.data);
            };

            eventSource.addEventListener('notice', function(event) {
                showNotification(event.data);
            });

            eventSource.onerror = function(event) {
                appendMessage('error', 'Error', '连接关闭');
                eventSource.close();
                if (eventSource.readyState === EventSource.CLOSED || eventSource.readyState === EventSource.CONNECTING) {
                    appendMessage('info', 'Info', '尝试重新连接...');
                    setTimeout(function() {
                        startSSE();  // 重连
                    }, 5000);  // 5 秒后重连
                }                
            };
            resolve(eventSource);
        });
    }

    // 发送消息,event为可选参数,如果不传表示发送message,否则发送指定event
    // 此处可改为向业务系统调用,由业务系统鉴权及检查消息内容后调用sse的send接口发送消息
    function send(event) {
        uid = $('#targetUid').val();
        if (uid == '') {
            alert('please input target uid');
            return;
        }
        data = $('#data').val();
        if (data == '') {
            alert('please input data');
            return;
        }
        $.ajax({
            url: '/send',
            method: 'POST',
            contentType: 'application/json',
            data: JSON.stringify({
                uid: uid,
                data: data,
                event: event
            }),
            success: function(response) {
                if(response.code != 1) {
                    alert('send message failed: [' + response.code + "] " + response.msg);
                    return;
                }
                else {
                    type = event ? event : 'message';
                    msg = 'send ' + type + ' to ' + uid + '\'s ' + response.result + ' device, data: ' +  $('#data').val();
                    appendMessage('info', 'Send', msg);
                    $('#data').val('');
                }
            },
            error: function() {
                alert('send message failed');
            }
        });
    }
    
    </script>
</head>
<body>
    <h1>SSE Demo</h1>
    <div id="user-form">
        <label for="uid">用户ID:</label>
        <input type="text" id="uid" name="uid" minlength="1" required>
        <button id="connect-sse" onclick="startSSE()">连接SSE</button>
    </div>
    <div id="send-form" style="display: none;">
        <label id="yourId" style="margin-left: 15px;"></label><br/>
        <div style="margin-bottom: 10px;"></div> 
        <label for="targetUid">对方的ID:</label>
        <input type="text" id="targetUid" name="targetUid" minlength="1" required>
        <div style="margin-bottom: 10px;"></div> 
        <label for="data">发送内容:</label>
        <textarea id="data" name="data" minlength="1" required></textarea>
        <div style="margin-bottom: 10px;"></div> 
        <button id="send-message" onclick="send()">发送消息</button>
        <button id="send-notification" onclick="send('notice')">发送通知</button>
    </div>

    <!-- 消息展示区域 -->
    <div id="messages"></div>

</body>
</html>