因为众所周知的原因,chrome的浏览器推送始终不能用。本文推荐一种使用SSE技术替代厂商通道实现全平台浏览器推送的方案。
原理
浏览器推送
SSE推送
增加成本
- redis,服务端需要单例或集群部署的redis实例(可与其他服务共用);
- sse-broker,一个或多个sse-broker实例,每个实例占用10M内存,每个连接可按2k内存计算;
步骤
安装运行sse-broker
- 下载sse-broker:Releases · sssxyd/go-sse-broker
- 安装,windows下解压,linux下 rpm -Uvh xxx.rpm
- 编辑config.toml windows下在解压所在目录,linux下在 /etc/sse-broker/config.toml
- 运行服务,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页面
- 复制附录中的 sse-demo.html 到你的web站点根目录下
- 浏览器访问:your.domain.com/sse-demo.ht…
- 输入uid并连接sse服务
- 向另一个uid发送通知或消息
对比
本方案以当前站点的私有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>