mirror of
https://github.com/labring/laf.git
synced 2025-12-08 20:26:22 +00:00
3.4 KiB
3.4 KiB
| title |
|---|
| 云函数处理 WebSocket 长连接 |
{{ $frontmatter.title }}
WebSocket 是一个长连接,可以主动给客户端推送消息,但是很可惜很多 Severless 架构不支持 WebSocket 的功能。值得开心的是,Laf 可以很轻松的使用 WebSocket,这给了我们更多的应用场景。
WebSocket 云函数
在 Laf 使用 WebSocket 需要创建一个云函数并且命名为 __websocket__
:::tip
WebSocket 云函数名为固定云函数名:__websocket__,其他名称均不会生效
:::
以下是云函数中处理 WebSocket 示例:
export async function main(ctx: FunctionContext) {
if (ctx.method === "WebSocket:connection") {
ctx.socket.send('hi connection succeed')
}
if (ctx.method === 'WebSocket:message') {
const { data } = ctx.params;
console.log(data.toString());
ctx.socket.send("I have received your message");
}
}
更多用法请参考: https://github.com/websockets/ws
客户端 WebSocket 连接
const wss = new WebSocket("wss://your-own-appid.laf.run/__websocket__");
wss.onopen = (socket) => {
console.log("connected");
wss.send("hi");
};
wss.onmessage = (res) => {
console.log("收到了新的信息......");
console.log(res.data);
};
wss.onclose = () => {
console.log("closed");
};
连接演示
1、替换 __websocket__ 代码并发布
import cloud from '@lafjs/cloud'
export async function main(ctx: FunctionContext) {
// 初始化 websocket user Map 列表
// 也可用数据库保存,本示例代码用的 Laf 云函数的全局缓存
let wsMap = cloud.shared.get("wsMap") // 获取 wsMap
if(!wsMap){
wsMap = new Map()
cloud.shared.set("wsMap", wsMap) // 设置 wsMap
}
// websocket 连接成功
if (ctx.method === "WebSocket:connection") {
const userId = generateUserId()
wsMap = cloud.shared.get("wsMap") // 获取 wsMap
wsMap.set(userId, ctx.socket);
cloud.shared.set("wsMap", wsMap) // 设置 wsMap
ctx.socket.send("连接成功,你的 userID 是:"+userId);
}
// websocket 消息事件
if (ctx.method === "WebSocket:message") {
const { data } = ctx.params;
console.log("接收到的信息:",data.toString());
const userId = getKeyByValue(wsMap, ctx.socket);
ctx.socket.send("服务端已接收到消息事件,你的 userID 是:"+userId);
}
// websocket 关闭消息
if (ctx.method === "WebSocket:close") {
wsMap = cloud.shared.get("wsMap") // 获取 wsMap
const userId = getKeyByValue(wsMap, ctx.socket);
wsMap.delete(userId);
cloud.shared.set("wsMap", wsMap) // 设置 wsMap
ctx.socket.send("服务端已接收到关闭事件消息,你的 userID 是:"+userId);
}
}
// 生成随机用户 ID
function generateUserId() {
return Math.random().toString(36).substring(2, 15);
}
// 遍历 userID
function getKeyByValue(map, value) {
for (const [key, val] of map.entries()) {
if (val === value) {
return key;
}
}
}
2、获得你的 WebSocket 链接
一般格式为:
your-own-appid 换成你的 Laf 应用 appid
wss://your-own-appid.laf.run/__websocket__
3、客户端链接 WebSocket 链接,并获得 userID
4、新建云函数通过 userID 去推送消息
import cloud from '@lafjs/cloud'
export async function main(ctx: FunctionContext) {
const userID = ''
let wsMap = cloud.shared.get("wsMap")
ctx.socket = wsMap.get(userID)
ctx.socket.send("消息测试");
}