mpush 是一套致力于用最简单,最快速的方式把消息从任何地方推送到指定的人的终端的系统,就像server 酱一样
只不过接收消息的不是微信而是任何可以建立 websocket 的客户端,就像下一点提到的安卓客户端,又或者是 http 服务器(webhook 方式),亦或者 WebPush
- fcm更新为使用http v1 api
- 添加了telegram bot作为接收端(未来加入发送)接入
- 2.1 的 FCM 接入改名为 WebPush socket 通信的 cmd 有所变化
- 新增 FCM 接入方式(2.1)
- 使用 typescript 重新编写
- 新增一对多按组推送
- 新增 webhook 的接入方式
- 新增 websocket 客户端双向传递消息
- 已经可以使用,功能正在完善中
- 2.x 版本 mpush 需要配合使用 2.x 版本的 mpush-android-client
- 支持 pwa
- 支持在安卓手机上使用 WebPush 推送(在安卓上为 fcm)
在发布页下载并解压 仅需要 ndoe 环境(建议使用 LTS 10.16.2 或以上版本)
npm install --production
# 按需修改完配置文件后运行
npm run start
git clone https://github.com/kooritea/mpush.git
cd mpush
npm install
# 按需修改完配置文件后运行
npm run dev
直接编辑根目录的 config.json 2.1 和 2.0 已有有较大幅度变化
{
"token": "test",
"http": {
"verifyToken": false, // 验证token,token位置为http头的authorization字段
"port": 9093,
"waitTimeout": 5000, //等待客户端答复的最长时间,超时后会返回目标客户端的接受状态
"cors": true, // 是否允许跨域请求
},
"websocket": {
"verifyToken": true, // websocket连接起始阶段的AUTH指令是否验证token
"port": 9093, // 可以和上面的http使用同一个端口,建议使用同一个端口
"authTimeout": 30000, // 时间内未验证成功将会断开socket
"retryTimeout": 3000, // 时间内未收到socket客户端的消息确认会重新发送
"waitTimeout": 5000 // 使用socket客户端向其他客户端发送消息时,等待其他客户端答复的最长时间,超时后会返回目标客户端的接受状态
},
"webhook": {
"retryTimeout": 3000,//时间内未收到2xx状态码会重新发送
"clients": [
{
"name": "wh1",
"group": "group2",
"url": "http://127.0.0.1:8089/",
"method": "GET", // 请求体下面补充
"proxy": "http://127.0.0.1:12333"// 代理设置,可以在clients内对单个客户端设置,也可以在webhook.proxy统一设置
}
],
"proxy": "http://127.0.0.1:12333"// 代理设置
},
"webpush": {
"apiKey": "", // FCM apiKey,获取方式下面补充
"proxy": "http://127.0.0.1:12333",// http代理
"retryTimeout": 10000 // 向WebPush服务器发送请求等待时间,超时未成功会重试
},
"fcm": {
"account": {},// firebase SDK的项目参数,获取方式见下面第四大点
"proxy": "http://127.0.0.1:62333",
"retryTimeout": 10000
},
"telegram": {
"botToken": string, // 找@botFather拿,拿到bot后回复/help 按提示认证和注册
"proxy": "http://127.0.0.1:62333",
"retryTimeout": 10000
}
}
具体接入方式的实现由客户端实现,仅需要在客户端中填写 name、group、token 即可
收到消息会以 http 请求的方式发送到指定的服务器,配置参考第二点中 webhook 字段
额外信息(extra)是指text,desp的字段外的字段
额外信息可以配合客户端实现例如: 优先级,scheme等功能
token 存放在 http 头的 authorization 字段
// get
// 所有额外信息都会平铺到url上,但额外信息如果是多层对象则不会继续平铺,例如下面的a字段和b字段,b是一个编码的JSON字符串{hello:'world'}
'/mpush?sendType=personal&target=wh1&fromMethod=websock&fromName=anonymous&mid=111111111111&text=text10&desp=desp10&a=233&b=%7B%22hello%22:%22world%22%7D'
// post
// 这里的额外信息会全部放到message.extra字段,没有额外信息字段的时候extra为空对象
{
cmd: 'MESSAGE',
data: {
sendType: "personal" | "group",
target: name | group name,
from: {
method: 'websocket' | 'http',
name: name | ''
},
mid: timestamp,
message: {
text: string,
desp: string,
extra: {
a: 233,
b: {
hello: 'world'
}
}
}
}
}
接收到 webhook 请求后 需要返回 2xx的响应码,否则超过设置的时间后会重新发送该消息
(1) 使用 http GET 请求发送消息,GET 方法接收 text 和 desp 两个字段,text 一般用作 title,参考 server 酱,除了 text 和 desp 参数,其他参数都会被放到 extra 字段中返回给接收方
格式
curl http://${HOST}:${HTTP_PORT}/${name}.${type:'send' | 'group'}?text=${text}&desp=${desp}&a=233&b=%7B%22hello%22:%22world%22%7D
其中type
为send
的时候 name 为客户端的 name
为group
的时候name
为group name
,即一对多推送
例如
向 name 为kooritea
的客户端推送消息 text 为hello
,desp 为world
的消息
curl http://HOST:HTTP_PORT/kooritea.send?text=hello&desp=world
向所有属于kgroup
组的客户端发送消息
curl http://HOST:HTTP_PORT/kgroup.group?text=hello&desp=world
{
"cmd": "MESSAGE",
"data" : {
"sendType": "personal" | "group",
"target": string,
"message": {
"text": string,
"desp": string,
"extra": {
// 任意字段,由接收方自行定义,例如android客户端可以接收一个scheme作为点击通知时跳转的应用、telegram客户端接收一个parse_mode决定文本的解析方式
}
}
}
}
// response
{
cmd: 'MESSAGE_REPLY',
data: {
"client1 name": "ready",
"client2 name": "ok",
"client3 name": "no",
"client4 name": "wait",
"client5 name": "webpush-wait",
"client5 name": "webpush-ok",
}
}
status | 意义 |
---|---|
ready | 正在送信队列排队 |
ok | 送达,对端已确认 |
no | 推送失败,没有找到该客户端(只会出现在一对一推送) |
wait | 已发送,等待对端确认 |
webpush-wait | 已通过 WebPush 发送,等待对端确认 |
webpush-send | 已发送 WebPush 消息到 Google(不代表消息送达) |
webpush-ok | 客户端确认 WebPush 消息送达 |
这部分会由客户端实现(具体实现方式可以看下面的开发文档),用户只需要选择目标和内容
-
登录 firebase 控制台
https://console.firebase.google.com/ -
创建一个项目
-
进入项目 添加应用 -> 添加后在上面的 tab 的服务帐号 -> 生成新的私钥
长这样
{
type: string;
project_id: string;
private_key_id: string;
private_key: string;
client_email: string;
client_id: string;
auth_uri: string;
token_uri: string;
auth_provider_x509_cert_url: string;
client_x509_cert_url: string;
universe_domain: string;
}
整个放到配置文件的fcm.account
请使用 nginx 等反代服务器
下面是 nginx 的示例配置
http {
...
server {
listen 9094 ssl;
server_name your.domain.com;
ssl_certificate "/etc/nginx/certs/your.domain.com.cer";
ssl_certificate_key "/etc/nginx/certs/your.domain.com.key";
ssl_session_cache shared:SSL:32768;
ssl_session_timeout 10m;
ssl_ciphers HIGH:!aNULL:!MD5;
ssl_prefer_server_ciphers on;
ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3;
server_tokens off;
location /mpush {
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $host;
proxy_read_timeout 300;
#mpush websocket path and port
proxy_pass http://127.0.0.1:9094;
}
}
}
// npm install ws --save
const WebSocket = require('ws')
const server = 'wss://domain.example/path'
const token = 'your token'
const name = 'your name'
const group = 'your group'
const ws = new WebSocket(server);
ws.on('error', console.error);
ws.on('open', function open() {
ws.send(JSON.stringify({
cmd: "AUTH",
data: {
token,
name,
group
}
}));
setInterval(()=>{
ws.ping()
}, 60000)
});
ws.on('message', function message(data) {
const packet = JSON.parse(data)
switch(packet.cmd){
case 'MESSAGE':{
ws.send(JSON.stringify({
cmd: "MESSAGE_CALLBACK",
data: {
mid: packet.data.mid
}
}))
onMessage(packet)
break
}
default:{
console.log(packet)
}
}
});
function onMessage(packet){
console.log(packet)
}