feat: Update code formatting and fix minor issues
All checks were successful
Egg CI/CD / build-image (push) Successful in 8m16s
Egg CI/CD / deploy (push) Successful in 7m47s

This commit is contained in:
zhaoyingbo 2024-05-05 02:34:16 +00:00
parent 7b3565926b
commit 865308ee31
31 changed files with 1293 additions and 1293 deletions

View File

@ -1,31 +1,31 @@
{
"name": "egg_server",
"image": "micr.cloud.mioffice.cn/zhaoyingbo/dev:bun",
"remoteUser": "bun",
"containerUser": "bun",
"forwardPorts": [3000],
"customizations": {
"vscode": {
"settings": {
"files.autoSave": "afterDelay",
"editor.guides.bracketPairs": true,
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true
},
"extensions": [
"dbaeumer.vscode-eslint",
"esbenp.prettier-vscode",
"eamodio.gitlens",
"unifiedjs.vscode-mdx",
"litiany4.umijs-plugin-model",
"oderwat.indent-rainbow",
"jock.svg",
"ChakrounAnas.turbo-console-log",
"Gruntfuggly.todo-tree",
"MS-CEINTL.vscode-language-pack-zh-hans",
"Alibaba-Cloud.tongyi-lingma"
]
}
},
"postCreateCommand": "bash -i /workspaces/egg_server/.devcontainer/initial.bash"
}
{
"name": "egg_server",
"image": "micr.cloud.mioffice.cn/zhaoyingbo/dev:bun",
"remoteUser": "bun",
"containerUser": "bun",
"forwardPorts": [3000],
"customizations": {
"vscode": {
"settings": {
"files.autoSave": "afterDelay",
"editor.guides.bracketPairs": true,
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true
},
"extensions": [
"dbaeumer.vscode-eslint",
"esbenp.prettier-vscode",
"eamodio.gitlens",
"unifiedjs.vscode-mdx",
"litiany4.umijs-plugin-model",
"oderwat.indent-rainbow",
"jock.svg",
"ChakrounAnas.turbo-console-log",
"Gruntfuggly.todo-tree",
"MS-CEINTL.vscode-language-pack-zh-hans",
"GitHub.copilot"
]
}
},
"postCreateCommand": "bash -i /workspaces/egg_server/.devcontainer/initial.bash"
}

2
.gitattributes vendored
View File

@ -1 +1 @@
*.lockb binary diff=lockb
*.lockb binary diff=lockb

View File

@ -1,57 +1,57 @@
name: Egg CI/CD
on: [push]
jobs:
build-image:
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
- name: Check out repository code
uses: actions/checkout@v3
- name: Set up QEMU
uses: docker/setup-qemu-action@v2
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v2
- name: Login to Docker Hub
uses: docker/login-action@v2
with:
registry: git.yingbo.im:333
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and push
uses: docker/build-push-action@v4
with:
push: true
tags: git.yingbo.im:333/zhaoyingbo/egg_server:${{ github.sha }}
deploy:
needs: build-image
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
# 检出代码
- name: Check out repository code
uses: actions/checkout@v3
# 使用scp命令将docker-compose.yml文件上传到服务器
- name: Upload docker-compose.yml to server
uses: appleboy/scp-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
source: docker-compose.yml
target: /home/${{ secrets.SERVER_USERNAME }}/docker/egg_server
# 登录服务器执行docker-compose命令
- name: Login to the server and execute docker-compose command
uses: appleboy/ssh-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
script: |
docker login -u ${{ secrets.DOCKER_USERNAME }} -p ${{ secrets.DOCKER_PASSWORD }} git.yingbo.im:333
cd /home/${{ secrets.SERVER_USERNAME }}/docker/egg_server
sed -i "s/sha/${{ github.sha }}/g" docker-compose.yml
docker compose up -d --force-recreate --no-deps egg_server
name: Egg CI/CD
on: [push]
jobs:
build-image:
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
- name: Check out repository code
uses: actions/checkout@v3
- name: Set up QEMU
uses: docker/setup-qemu-action@v2
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v2
- name: Login to Docker Hub
uses: docker/login-action@v2
with:
registry: git.yingbo.im:333
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and push
uses: docker/build-push-action@v4
with:
push: true
tags: git.yingbo.im:333/zhaoyingbo/egg_server:${{ github.sha }}
deploy:
needs: build-image
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
# 检出代码
- name: Check out repository code
uses: actions/checkout@v3
# 使用scp命令将docker-compose.yml文件上传到服务器
- name: Upload docker-compose.yml to server
uses: appleboy/scp-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
source: docker-compose.yml
target: /home/${{ secrets.SERVER_USERNAME }}/docker/egg_server
# 登录服务器执行docker-compose命令
- name: Login to the server and execute docker-compose command
uses: appleboy/ssh-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
script: |
docker login -u ${{ secrets.DOCKER_USERNAME }} -p ${{ secrets.DOCKER_PASSWORD }} git.yingbo.im:333
cd /home/${{ secrets.SERVER_USERNAME }}/docker/egg_server
sed -i "s/sha/${{ github.sha }}/g" docker-compose.yml
docker compose up -d --force-recreate --no-deps egg_server

116
.gitignore vendored
View File

@ -1,58 +1,58 @@
# Logs
logs
*.log
npm-debug.log*
# Runtime data
pids
*.pid
*.seed
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
# nyc test coverage
.nyc_output
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# node-waf configuration
.lock-wscript
# Compiled binary addons (http://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules
jspm_packages
# Optional npm cache directory
.npm
# Optional REPL history
.node_repl_history
# 0x
profile-*
# mac files
.DS_Store
# vim swap files
*.swp
# webstorm
.idea
# vscode
.vscode
*code-workspace
# clinic
profile*
*clinic*
*flamegraph*
# Logs
logs
*.log
npm-debug.log*
# Runtime data
pids
*.pid
*.seed
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
# nyc test coverage
.nyc_output
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# node-waf configuration
.lock-wscript
# Compiled binary addons (http://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules
jspm_packages
# Optional npm cache directory
.npm
# Optional REPL history
.node_repl_history
# 0x
profile-*
# mac files
.DS_Store
# vim swap files
*.swp
# webstorm
.idea
# vscode
.vscode
*code-workspace
# clinic
profile*
*clinic*
*flamegraph*

View File

@ -1,22 +1,22 @@
FROM micr.cloud.mioffice.cn/zhaoyingbo/bun:alpine
ENV TZ=Asia/Shanghai
RUN apk update \
&& apk add tzdata \
&& cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
&& echo "Asia/Shanghai" > /etc/timezone
WORKDIR /app
COPY package*.json ./
COPY bun.lockb ./
RUN bun install
COPY . .
EXPOSE 3000
FROM micr.cloud.mioffice.cn/zhaoyingbo/bun:alpine
ENV TZ=Asia/Shanghai
RUN apk update \
&& apk add tzdata \
&& cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
&& echo "Asia/Shanghai" > /etc/timezone
WORKDIR /app
COPY package*.json ./
COPY bun.lockb ./
RUN bun install
COPY . .
EXPOSE 3000
CMD ["bun", "run", "start"]

BIN
bun.lockb

Binary file not shown.

View File

@ -1,9 +1,9 @@
import messageGroup from "./messageGroup";
import tenantAccessToken from "./tenantAccessToken";
const db = {
messageGroup,
tenantAccessToken,
};
export default db;
import messageGroup from "./messageGroup";
import tenantAccessToken from "./tenantAccessToken";
const db = {
messageGroup,
tenantAccessToken,
};
export default db;

View File

@ -1,13 +1,13 @@
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
const getOne = (groupId: string) =>
managePb404(
async () => await pbClient.collection("message_group").getOne(groupId)
);
const messageGroup = {
getOne,
};
export default messageGroup;
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
const getOne = (groupId: string) =>
managePb404(
async () => await pbClient.collection("message_group").getOne(groupId)
);
const messageGroup = {
getOne,
};
export default messageGroup;

View File

@ -1,14 +1,14 @@
interface PBMessageGroup {
collectionId: string;
collectionName: string;
updated: string;
created: string;
desc: string;
id: string;
name: string;
email?: string[];
chat_id?: string[];
open_id?: string[];
union_id?: string[];
user_id?: string[];
}
interface PBMessageGroup {
collectionId: string;
collectionName: string;
updated: string;
created: string;
desc: string;
id: string;
name: string;
email?: string[];
chat_id?: string[];
open_id?: string[];
union_id?: string[];
user_id?: string[];
}

View File

@ -1,5 +1,5 @@
import PocketBase from 'pocketbase';
const pbClient = new PocketBase('https://eggpb.imoaix.cn')
export default pbClient;
import PocketBase from 'pocketbase';
const pbClient = new PocketBase('https://eggpb.imoaix.cn')
export default pbClient;

View File

@ -1,32 +1,32 @@
import pbClient from "../pbClient";
let token = "";
/**
* token
* @param {string} value token
*/
const update = async (value: string) => {
await pbClient.collection("config").update("ugel8f0cpk0rut6", { value });
token = value;
console.log("reset access token success", value);
};
/**
* token
* @returns {string} token
*/
const get = async () => {
if (token) return token;
const { value } = await pbClient
.collection("config")
.getOne("ugel8f0cpk0rut6");
return value as string;
};
const tenantAccessToken = {
update,
get,
};
export default tenantAccessToken;
import pbClient from "../pbClient";
let token = "";
/**
* token
* @param {string} value token
*/
const update = async (value: string) => {
await pbClient.collection("config").update("ugel8f0cpk0rut6", { value });
token = value;
console.log("reset access token success", value);
};
/**
* token
* @returns {string} token
*/
const get = async () => {
if (token) return token;
const { value } = await pbClient
.collection("config")
.getOne("ugel8f0cpk0rut6");
return value as string;
};
const tenantAccessToken = {
update,
get,
};
export default tenantAccessToken;

View File

@ -1,10 +1,10 @@
FROM node:18.17.1-alpine3.18
# 更换国内源
RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.tuna.tsinghua.edu.cn/g' /etc/apk/repositories
# 设置时区
RUN apk update \
&& apk add tzdata \
&& cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
FROM node:18.17.1-alpine3.18
# 更换国内源
RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.tuna.tsinghua.edu.cn/g' /etc/apk/repositories
# 设置时区
RUN apk update \
&& apk add tzdata \
&& cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime \
&& echo "Asia/Shanghai" > /etc/timezone

View File

@ -1,2 +1,2 @@
docker build -t micr.cloud.mioffice.cn/zhaoyingbo/node:18.17.1-alpine3.18 .
docker build -t micr.cloud.mioffice.cn/zhaoyingbo/node:18.17.1-alpine3.18 .
docker push micr.cloud.mioffice.cn/zhaoyingbo/node:18.17.1-alpine3.18

View File

@ -1,9 +1,9 @@
version: "3"
services:
egg_server:
image: git.yingbo.im:333/zhaoyingbo/egg_server:sha
container_name: egg_server
restart: always
ports:
- 3003:3000
version: "3"
services:
egg_server:
image: git.yingbo.im:333/zhaoyingbo/egg_server:sha
container_name: egg_server
restart: always
ports:
- 3003:3000

View File

@ -1,20 +1,20 @@
import { manageBotReq } from "./routes/bot";
import { manageMessageReq } from "./routes/message";
import { initSchedule } from "./schedule";
initSchedule()
Bun.serve({
async fetch(req) {
const url = new URL(req.url);
// 根路由
if (url.pathname === "/") return new Response("hello, glade to see you!");
// 机器人
if (url.pathname === '/bot') return await manageBotReq(req);
// 消息发送
if (url.pathname === '/message') return await manageMessageReq(req);
// 其他
return new Response('OK')
},
port: 3000
import { manageBotReq } from "./routes/bot";
import { manageMessageReq } from "./routes/message";
import { initSchedule } from "./schedule";
initSchedule()
Bun.serve({
async fetch(req) {
const url = new URL(req.url);
// 根路由
if (url.pathname === "/") return new Response("hello, glade to see you!");
// 机器人
if (url.pathname === '/bot') return await manageBotReq(req);
// 消息发送
if (url.pathname === '/message') return await manageMessageReq(req);
// 其他
return new Response('OK')
},
port: 3000
});

View File

@ -1,19 +1,19 @@
{
"name": "egg_server",
"module": "index.ts",
"type": "module",
"scripts": {
"start": "bun run index.ts"
},
"devDependencies": {
"bun-types": "latest"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"dependencies": {
"@types/node-schedule": "^2.1.6",
"node-schedule": "^2.1.1",
"pocketbase": "^0.21.1"
}
{
"name": "egg_server",
"module": "index.ts",
"type": "module",
"scripts": {
"start": "bun run index.ts"
},
"devDependencies": {
"bun-types": "latest"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"dependencies": {
"@types/node-schedule": "^2.1.6",
"node-schedule": "^2.1.1",
"pocketbase": "^0.21.1"
}
}

View File

@ -1,58 +1,58 @@
import { sleep } from "bun";
import { fetchCIMonitor } from "../../service";
import { getActionType, getIsActionMsg } from "../../utils/msgTools";
import { updateCard } from "../../utils/sendMsg";
const makeChatIdCard = async (body: LarkUserAction) => {
await sleep(500);
return JSON.stringify({
type: "template",
data: {
config: {
update_multi: true,
},
template_id: "ctp_AAi3NnHb6zgK",
template_variable: {
chat_id: body.open_chat_id,
},
},
});
};
const ACTION_MAP = {
chat_id: makeChatIdCard,
ci: fetchCIMonitor,
};
/**
*
* @param {LarkUserAction} body
*/
const manageBtnClick = async (body: LarkUserAction) => {
const { action } = body?.action?.value as {
action: keyof typeof ACTION_MAP;
};
if (!action) return;
const func = ACTION_MAP[action];
if (!func) return;
const card = await func(body);
if (!card) return;
// 更新飞书的卡片
await updateCard(body.open_message_id, card);
};
/**
* Action消息
* @param {LarkUserAction} body
*/
export const manageActionMsg = (body: LarkUserAction) => {
// 过滤非Action消息
if (!getIsActionMsg(body)) {
return false;
}
const actionType = getActionType(body);
if (actionType === "button") {
manageBtnClick(body);
}
return true;
};
import { sleep } from "bun";
import { fetchCIMonitor } from "../../service";
import { getActionType, getIsActionMsg } from "../../utils/msgTools";
import { updateCard } from "../../utils/sendMsg";
const makeChatIdCard = async (body: LarkUserAction) => {
await sleep(500);
return JSON.stringify({
type: "template",
data: {
config: {
update_multi: true,
},
template_id: "ctp_AAi3NnHb6zgK",
template_variable: {
chat_id: body.open_chat_id,
},
},
});
};
const ACTION_MAP = {
chat_id: makeChatIdCard,
ci: fetchCIMonitor,
};
/**
*
* @param {LarkUserAction} body
*/
const manageBtnClick = async (body: LarkUserAction) => {
const { action } = body?.action?.value as {
action: keyof typeof ACTION_MAP;
};
if (!action) return;
const func = ACTION_MAP[action];
if (!func) return;
const card = await func(body);
if (!card) return;
// 更新飞书的卡片
await updateCard(body.open_message_id, card);
};
/**
* Action消息
* @param {LarkUserAction} body
*/
export const manageActionMsg = (body: LarkUserAction) => {
// 过滤非Action消息
if (!getIsActionMsg(body)) {
return false;
}
const actionType = getActionType(body);
if (actionType === "button") {
manageBtnClick(body);
}
return true;
};

View File

@ -1,98 +1,98 @@
import { getChatId, getIsEventMsg, getMsgType } from "../../utils/msgTools";
import { sendMsg } from "../../utils/sendMsg";
/**
*
* @param {LarkMessageEvent} body
* @returns {string}
*/
export const getMsgText = (body: LarkMessageEvent) => {
// TODO: 如果之后想支持单独提醒,这里需要做模板解析
try {
const { text }: { text: string } = JSON.parse(
body?.event?.message?.content
);
// 去掉@_user_1相关的内容例如 '@_user_1 测试' -> '测试'
const textWithoutAt = text.replace(/@_user_\d+/g, "");
// 去除空格和换行
const textWithoutSpace = textWithoutAt.replace(/[\s\n]/g, "");
return textWithoutSpace;
} catch (e) {
return "";
}
};
/**
*
* @param {LarkMessageEvent} body
* @returns {boolean}
*/
const filterIllegalMsg = (body: LarkMessageEvent) => {
const chatId = getChatId(body);
if (!chatId) return true;
// 获取msgType
const msgType = getMsgType(body);
// 发表情包就直接发回去
if (msgType === "sticker") {
const content = body?.event?.message?.content;
sendMsg("chat_id", chatId, "sticker", content);
return true;
}
// 剩下的非文字消息暂时不处理
if (msgType !== "text") {
const textList = [
"仅支持普通文本内容[黑脸]",
"唔...我只能处理普通文本哦[泣不成声]",
"噢!这似乎是个非普通文本[看]",
"哇!这是什么东东?我只懂普通文本啦![可爱]",
"只能处理普通文本内容哦[捂脸]",
];
const content = JSON.stringify({
text: textList[Math.floor(Math.random() * textList.length)],
});
sendMsg("chat_id", chatId, "text", content);
return true;
}
// 还得过滤下艾特全体成员的消息
if (getMsgText(body).includes("@_all")) {
return true;
}
return false;
};
/**
*
* @param {LarkMessageEvent} body
*/
const replyNomalMsg = async (body: LarkMessageEvent) => {
const chatId = getChatId(body);
const content = JSON.stringify({
type: "template",
data: {
config: {
enable_forward: false,
update_multi: true,
},
template_id: "ctp_AAyVx5R39xU9",
},
});
await sendMsg("chat_id", chatId, "interactive", content);
};
/**
* Event消息
* @param {LarkUserAction} body
*/
export const manageEventMsg = (body: LarkMessageEvent) => {
// 过滤非Event消息
if (!getIsEventMsg(body)) {
return false;
}
// 过滤非法消息
if (filterIllegalMsg(body)) {
return true;
}
// 临时返回消息
replyNomalMsg(body);
return true;
};
import { getChatId, getIsEventMsg, getMsgType } from "../../utils/msgTools";
import { sendMsg } from "../../utils/sendMsg";
/**
*
* @param {LarkMessageEvent} body
* @returns {string}
*/
export const getMsgText = (body: LarkMessageEvent) => {
// TODO: 如果之后想支持单独提醒,这里需要做模板解析
try {
const { text }: { text: string } = JSON.parse(
body?.event?.message?.content
);
// 去掉@_user_1相关的内容例如 '@_user_1 测试' -> '测试'
const textWithoutAt = text.replace(/@_user_\d+/g, "");
// 去除空格和换行
const textWithoutSpace = textWithoutAt.replace(/[\s\n]/g, "");
return textWithoutSpace;
} catch (e) {
return "";
}
};
/**
*
* @param {LarkMessageEvent} body
* @returns {boolean}
*/
const filterIllegalMsg = (body: LarkMessageEvent) => {
const chatId = getChatId(body);
if (!chatId) return true;
// 获取msgType
const msgType = getMsgType(body);
// 发表情包就直接发回去
if (msgType === "sticker") {
const content = body?.event?.message?.content;
sendMsg("chat_id", chatId, "sticker", content);
return true;
}
// 剩下的非文字消息暂时不处理
if (msgType !== "text") {
const textList = [
"仅支持普通文本内容[黑脸]",
"唔...我只能处理普通文本哦[泣不成声]",
"噢!这似乎是个非普通文本[看]",
"哇!这是什么东东?我只懂普通文本啦![可爱]",
"只能处理普通文本内容哦[捂脸]",
];
const content = JSON.stringify({
text: textList[Math.floor(Math.random() * textList.length)],
});
sendMsg("chat_id", chatId, "text", content);
return true;
}
// 还得过滤下艾特全体成员的消息
if (getMsgText(body).includes("@_all")) {
return true;
}
return false;
};
/**
*
* @param {LarkMessageEvent} body
*/
const replyNomalMsg = async (body: LarkMessageEvent) => {
const chatId = getChatId(body);
const content = JSON.stringify({
type: "template",
data: {
config: {
enable_forward: false,
update_multi: true,
},
template_id: "ctp_AAyVx5R39xU9",
},
});
await sendMsg("chat_id", chatId, "interactive", content);
};
/**
* Event消息
* @param {LarkUserAction} body
*/
export const manageEventMsg = (body: LarkMessageEvent) => {
// 过滤非Event消息
if (!getIsEventMsg(body)) {
return false;
}
// 过滤非法消息
if (filterIllegalMsg(body)) {
return true;
}
// 临时返回消息
replyNomalMsg(body);
return true;
};

View File

@ -1,15 +1,15 @@
import { manageActionMsg } from "./activeMsg";
import { manageEventMsg } from "./eventMsg";
export const manageBotReq = async (req: Request) => {
const body = (await req.json()) as any;
// 验证机器人
if (body?.type === "url_verification") {
return Response.json({ challenge: body?.challenge });
}
// 处理Event消息
if (manageEventMsg(body)) return new Response("success");
// 处理Action消息
if (manageActionMsg(body)) return new Response("success");
return new Response("hello, glade to see you!");
};
import { manageActionMsg } from "./activeMsg";
import { manageEventMsg } from "./eventMsg";
export const manageBotReq = async (req: Request) => {
const body = (await req.json()) as any;
// 验证机器人
if (body?.type === "url_verification") {
return Response.json({ challenge: body?.challenge });
}
// 处理Event消息
if (manageEventMsg(body)) return new Response("success");
// 处理Action消息
if (manageActionMsg(body)) return new Response("success");
return new Response("hello, glade to see you!");
};

View File

@ -1,122 +1,122 @@
import db from "../../db";
import { sendMsg } from "../../utils/sendMsg";
interface BaseMsg {
msg_type: MsgType;
content: string;
}
interface GroupMsg extends BaseMsg {
group_id: string;
}
interface NormalMsg extends BaseMsg {
receive_id: string;
receive_id_type: ReceiveIDType;
}
type MessageReqJson = GroupMsg & NormalMsg;
const validateMessageReq = (body: MessageReqJson) => {
if (!body.group_id && !body.receive_id) {
return new Response("group_id or receive_id is required");
}
if (body.receive_id && !body.receive_id_type) {
return new Response("receive_id_type is required");
}
if (!body.msg_type) {
return new Response("msg_type is required");
}
if (!body.content) {
return new Response("content is required");
}
return false;
};
export const manageMessageReq = async (req: Request) => {
const body = (await req.json()) as MessageReqJson;
// 校验参数
const validateRes = validateMessageReq(body);
if (validateRes) return validateRes;
// 遍历所有id发送消息保存所有对应的messageId
const sendRes = {
chat_id: {} as Record<string, any>,
open_id: {} as Record<string, any>,
union_id: {} as Record<string, any>,
user_id: {} as Record<string, any>,
email: {} as Record<string, any>,
};
// 发送消息列表
const sendList = [] as Promise<any>[];
// 处理消息内容
const finalContent =
typeof body.content !== "string"
? JSON.stringify(body.content)
: body.content;
if (body.group_id) {
// 获取所有接收者
const group = (await db.messageGroup.getOne(
body.group_id!
)) as PBMessageGroup;
if (!group) {
return new Response("group not found");
}
const { chat_id, open_id, union_id, user_id, email } = group;
// 构造发送消息函数
const makeSendFunc = (receive_id_type: ReceiveIDType) => {
return (receive_id: string) => {
sendList.push(
sendMsg(
receive_id_type,
receive_id,
body.msg_type,
finalContent
).then((res) => {
sendRes[receive_id_type][receive_id] = res;
})
);
};
};
// 创建消息列表
if (chat_id) chat_id.map(makeSendFunc("chat_id"));
if (open_id) open_id.map(makeSendFunc("open_id"));
if (union_id) union_id.map(makeSendFunc("union_id"));
if (user_id) user_id.map(makeSendFunc("user_id"));
if (email) email.map(makeSendFunc("email"));
}
if (body.receive_id && body.receive_id_type) {
sendList.push(
sendMsg(
body.receive_id_type,
body.receive_id,
body.msg_type,
finalContent
).then((res) => {
sendRes[body.receive_id_type][body.receive_id] = res;
})
);
}
try {
await Promise.all(sendList);
return Response.json({
code: 200,
msg: "ok",
data: sendRes,
});
} catch {
return Response.json({
code: 400,
msg: "send msg failed",
data: sendRes,
});
}
};
import db from "../../db";
import { sendMsg } from "../../utils/sendMsg";
interface BaseMsg {
msg_type: MsgType;
content: string;
}
interface GroupMsg extends BaseMsg {
group_id: string;
}
interface NormalMsg extends BaseMsg {
receive_id: string;
receive_id_type: ReceiveIDType;
}
type MessageReqJson = GroupMsg & NormalMsg;
const validateMessageReq = (body: MessageReqJson) => {
if (!body.group_id && !body.receive_id) {
return new Response("group_id or receive_id is required");
}
if (body.receive_id && !body.receive_id_type) {
return new Response("receive_id_type is required");
}
if (!body.msg_type) {
return new Response("msg_type is required");
}
if (!body.content) {
return new Response("content is required");
}
return false;
};
export const manageMessageReq = async (req: Request) => {
const body = (await req.json()) as MessageReqJson;
// 校验参数
const validateRes = validateMessageReq(body);
if (validateRes) return validateRes;
// 遍历所有id发送消息保存所有对应的messageId
const sendRes = {
chat_id: {} as Record<string, any>,
open_id: {} as Record<string, any>,
union_id: {} as Record<string, any>,
user_id: {} as Record<string, any>,
email: {} as Record<string, any>,
};
// 发送消息列表
const sendList = [] as Promise<any>[];
// 处理消息内容
const finalContent =
typeof body.content !== "string"
? JSON.stringify(body.content)
: body.content;
if (body.group_id) {
// 获取所有接收者
const group = (await db.messageGroup.getOne(
body.group_id!
)) as PBMessageGroup;
if (!group) {
return new Response("group not found");
}
const { chat_id, open_id, union_id, user_id, email } = group;
// 构造发送消息函数
const makeSendFunc = (receive_id_type: ReceiveIDType) => {
return (receive_id: string) => {
sendList.push(
sendMsg(
receive_id_type,
receive_id,
body.msg_type,
finalContent
).then((res) => {
sendRes[receive_id_type][receive_id] = res;
})
);
};
};
// 创建消息列表
if (chat_id) chat_id.map(makeSendFunc("chat_id"));
if (open_id) open_id.map(makeSendFunc("open_id"));
if (union_id) union_id.map(makeSendFunc("union_id"));
if (user_id) user_id.map(makeSendFunc("user_id"));
if (email) email.map(makeSendFunc("email"));
}
if (body.receive_id && body.receive_id_type) {
sendList.push(
sendMsg(
body.receive_id_type,
body.receive_id,
body.msg_type,
finalContent
).then((res) => {
sendRes[body.receive_id_type][body.receive_id] = res;
})
);
}
try {
await Promise.all(sendList);
return Response.json({
code: 200,
msg: "ok",
data: sendRes,
});
} catch {
return Response.json({
code: 400,
msg: "send msg failed",
data: sendRes,
});
}
};

View File

@ -1 +1 @@
# 批量发送消息,给已经订阅的用户和群组发送消息
# 批量发送消息,给已经订阅的用户和群组发送消息

View File

@ -1,20 +1,20 @@
import db from "../db"
export const resetAccessToken = async () => {
const URL = 'https://open.f.mioffice.cn/open-apis/auth/v3/tenant_access_token/internal'
const app_id = 'cli_a1eff35b43b89063'
const app_secret = 'IFSl8ig5DMwMnFjwPiljCfoEWlgRwDxW'
const res = await fetch(URL, {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({
app_id,
app_secret
})
})
const { tenant_access_token } = await res.json() as any
await db.tenantAccessToken.update(tenant_access_token)
return tenant_access_token
import db from "../db"
export const resetAccessToken = async () => {
const URL = 'https://open.f.mioffice.cn/open-apis/auth/v3/tenant_access_token/internal'
const app_id = 'cli_a1eff35b43b89063'
const app_secret = 'IFSl8ig5DMwMnFjwPiljCfoEWlgRwDxW'
const res = await fetch(URL, {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify({
app_id,
app_secret
})
})
const { tenant_access_token } = await res.json() as any
await db.tenantAccessToken.update(tenant_access_token)
return tenant_access_token
}

View File

@ -1,9 +1,9 @@
import { resetAccessToken } from "./accessToken";
import schedule from 'node-schedule'
export const initSchedule = async () => {
// 定时任务每15分钟刷新一次token
schedule.scheduleJob('*/15 * * * *', resetAccessToken);
// 立即执行一次
resetAccessToken()
}
import { resetAccessToken } from "./accessToken";
import schedule from 'node-schedule'
export const initSchedule = async () => {
// 定时任务每15分钟刷新一次token
schedule.scheduleJob('*/15 * * * *', resetAccessToken);
// 立即执行一次
resetAccessToken()
}

View File

@ -1,8 +1,8 @@
export const fetchCIMonitor = async () => {
try {
const res = await fetch("https://ci-monitor.xiaomiwh.cn/ci");
return ((await res.json()) as string) || "";
} catch {
return "";
}
};
export const fetchCIMonitor = async () => {
try {
const res = await fetch("https://ci-monitor.xiaomiwh.cn/ci");
return ((await res.json()) as string) || "";
} catch {
return "";
}
};

10
test.ts
View File

@ -1,5 +1,5 @@
console.log(
JSON.stringify({
text: "hello",
})
);
console.log(
JSON.stringify({
text: "hello",
})
);

View File

@ -1,21 +1,21 @@
[
{
"_id": "a5cb7b58-0c22-41fb-bc00-af460ef90c1b",
"colId": "history",
"containerId": "",
"name": "https://self.imoaix.cn/bot",
"url": "https://self.imoaix.cn/bot",
"method": "POST",
"sortNum": 0,
"created": "2023-08-15T07:47:05.439Z",
"modified": "2023-08-15T07:47:34.510Z",
"headers": [],
"params": [],
"body": {
"type": "json",
"raw": "{\r\n \"challenge\": \"c14206ad-5b5f-4a77-82b6-b39b7365392b\",\r\n \"token\": \"tV9djUKSjzVnekV7xTg2Od06NFTcsBnj\",\r\n \"type\": \"url_verification\"\r\n}",
"form": []
},
"tests": []
}
[
{
"_id": "a5cb7b58-0c22-41fb-bc00-af460ef90c1b",
"colId": "history",
"containerId": "",
"name": "https://self.imoaix.cn/bot",
"url": "https://self.imoaix.cn/bot",
"method": "POST",
"sortNum": 0,
"created": "2023-08-15T07:47:05.439Z",
"modified": "2023-08-15T07:47:34.510Z",
"headers": [],
"params": [],
"body": {
"type": "json",
"raw": "{\r\n \"challenge\": \"c14206ad-5b5f-4a77-82b6-b39b7365392b\",\r\n \"token\": \"tV9djUKSjzVnekV7xTg2Od06NFTcsBnj\",\r\n \"type\": \"url_verification\"\r\n}",
"form": []
},
"tests": []
}
]

View File

@ -1,22 +1,22 @@
{
"compilerOptions": {
"lib": ["ESNext"],
"module": "esnext",
"target": "esnext",
"moduleResolution": "bundler",
"moduleDetection": "force",
"allowImportingTsExtensions": true,
"noEmit": true,
"composite": true,
"strict": true,
"downlevelIteration": true,
"skipLibCheck": true,
"jsx": "react-jsx",
"allowSyntheticDefaultImports": true,
"forceConsistentCasingInFileNames": true,
"allowJs": true,
"types": [
"bun-types" // add Bun global
]
}
}
{
"compilerOptions": {
"lib": ["ESNext"],
"module": "esnext",
"target": "esnext",
"moduleResolution": "bundler",
"moduleDetection": "force",
"allowImportingTsExtensions": true,
"noEmit": true,
"composite": true,
"strict": true,
"downlevelIteration": true,
"skipLibCheck": true,
"jsx": "react-jsx",
"allowSyntheticDefaultImports": true,
"forceConsistentCasingInFileNames": true,
"allowJs": true,
"types": [
"bun-types" // add Bun global
]
}
}

902
typings.d.ts vendored
View File

@ -1,451 +1,451 @@
/**
*
*/
interface User {
/**
* id
*/
id: string;
/**
*
* @example zhaoyingbo
*/
userId: string;
/**
* open_id
*/
openId: string;
/**
*
*/
remindList: string[];
}
/**
*
*/
interface Remind {
/**
* id
*/
id: string;
/**
* id
*/
owner: string;
/**
* Id
*/
messageId: string;
/**
*
*/
subscriberType: "open_id" | "user_id" | "union_id" | "email" | "chat_id";
/**
* Id
*/
subscriberId: string;
/**
*
*/
needReply: boolean;
/**
*
*/
delayTime: number;
/**
*
*/
cardInfo: {
/**
*
*/
title: string;
/**
* key
*/
imageKey?: string;
/**
*
*/
content?: string;
/**
*
*/
confirmText?: string;
/**
*
*/
cancelText?: string;
/**
*
*/
delayText?: string;
} | null;
/**
*
*/
templateInfo: {
/**
* ID
* ${owner}
* ${remindTime}
*/
pendingTemplateId: string;
/**
* ID
* ${owner}
* ${remindTime}
* ${result} textresult对应的
* ${interactTime}
*/
interactedTemplateId: string;
/**
* ID
*/
confirmedTemplateId: string;
/**
* ID
*/
cancelededTemplateId: string;
/**
* ID
*/
delayedTemplateId: string;
} | null;
/**
*
*/
remindTimes: RemindTime[];
/**
*
*/
enabled: boolean;
/**
* yyyy-MM-dd HH:mm
*/
nextRemindTime: string;
/**
*
*/
nextRemindTimeCHS: string;
}
/**
*
*
*/
interface RemindTime {
/**
*
* single: 一次性
* daily: 每天
* weekly: 每周
* monthly: 每月
* yearly: 每年
* workday: 工作日
* holiday: 节假日
*/
frequency:
| "single"
| "daily"
| "weekly"
| "monthly"
| "yearly"
| "workday"
| "holiday";
/**
* 格式为HH:mm single类型时仅作展示用yyyy-MM-dd HH:mm
*/
time: string;
/**
* [1-7]frequency为weekly时有效
*/
daysOfWeek: number[];
/**
* [1-31]frequency为monthly时有效
*/
daysOfMonth: number[];
/**
* frequency为 yearly MM-dd
*/
dayOfYear: string;
}
/**
*
*
*/
interface RemindRecord {
/**
* Id
*/
id: string;
/**
* Id
*/
remindId: string;
/**
* Id
*/
messageId: string;
/**
*
* pending: 待确认
* delay: 已延迟
* confirmed: 已确认
* canceled: 已取消
*/
status: "pending" | "delayed" | "confirmed" | "canceled";
/**
* yyyy-MM-dd HH:mm
*/
remindTime: string;
/**
* yyyy-MM-dd HH:mm
*/
interactTime: string;
/**
* 07:00
*/
result: object;
}
/**
*
*/
interface Header {
/**
* ID
* @example 0f8ab23b60993cf8dd15c8cde4d7b0f5
*/
event_id: string;
/**
* token
* @example tV9djUKSjzVnekV7xTg2Od06NFTcsBnj
*/
token: string;
/**
*
* @example 1693565712117
*/
create_time: string;
/**
*
* @example im.message.receive_v1
*/
event_type: string;
/**
* tenant_key
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
/**
* app_id
* @example cli_a1eff35b43b89063
*/
app_id: string;
}
/**
* ID信息
*/
interface UserIdInfo {
/**
*
* @example ou_032f507d08f9a7f28b042fcd086daef5
*/
open_id: string;
/**
*
* @example on_7111660fddd8302ce47bf1999147c011
*/
union_id: string;
/**
*
* @example zhaoyingbo
*/
user_id: string;
}
/**
* AT的人的信息
*/
interface Mention {
/**
* ID信息
*/
id: UserIdInfo;
/**
*
* @example "@_user_1"
*/
key: string;
/**
*
* @example
*/
name: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
}
/**
*
*/
interface Message {
/**
* ID
* @example oc_433b1cb7a9dbb7ebe70a4e1a59cb8bb1
*/
chat_id: string;
/**
*
* @example group | p2p
*/
chat_type: string;
/**
* JSON字符串文本内容
* @example "{\"text\":\"@_user_1 测试\"}"
*/
content: string;
/**
*
* @example 1693565711996
*/
create_time: string;
/**
*
*/
mentions?: Mention[];
/**
* ID
* @example om_038fc0eceed6224a1abc1cdaa4266405
*/
message_id: string;
/**
*
* @example textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
*/
message_type: string;
}
/**
*
*/
interface Sender {
/**
* id
*/
sender_id: UserIdInfo;
/**
*
* @example user
*/
sender_type: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
}
/**
*
*/
interface Event {
message: Message;
sender: Sender;
}
/**
*
*/
interface LarkMessageEvent {
/**
*
* @example 2.0
*/
schema: string;
/**
*
*/
header: Header;
/**
*
*/
event: Event;
}
/**
* Action信息
*/
interface LarkUserAction {
/**
* open_id
*/
open_id: string;
/**
*
* @example zhaoyingbo
*/
user_id: string;
/**
* ID
* @example om_038fc0eceed6224a1abc1cdaa4266405
*/
open_message_id: string;
/**
* ID
* @example oc_433b1cb7a9dbb7ebe70a4e1a59cb8bb1
*/
open_chat_id: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
/**
* token
* @example tV9djUKSjzVnekV7xTg2Od06NFTcsBnj
*/
token: string;
/**
*
*/
action: {
/**
*
*/
value: any;
/**
*
* @example picker_datetime
*/
tag: string;
/**
*
* @example 2023-09-03 10:35 +0800
*/
option: string;
/**
*
*/
timezone: string;
};
}
type ReceiveIDType = "open_id" | "user_id" | "union_id" | "email" | "chat_id";
type MsgType = "text" | "post" | "image" | "file" | "audio" | "media" | "sticker" | "interactive" | "share_chat" | "share_user";
interface ServerResponse {
code: number;
data: any;
msg: string;
}
/**
*
*/
interface User {
/**
* id
*/
id: string;
/**
*
* @example zhaoyingbo
*/
userId: string;
/**
* open_id
*/
openId: string;
/**
*
*/
remindList: string[];
}
/**
*
*/
interface Remind {
/**
* id
*/
id: string;
/**
* id
*/
owner: string;
/**
* Id
*/
messageId: string;
/**
*
*/
subscriberType: "open_id" | "user_id" | "union_id" | "email" | "chat_id";
/**
* Id
*/
subscriberId: string;
/**
*
*/
needReply: boolean;
/**
*
*/
delayTime: number;
/**
*
*/
cardInfo: {
/**
*
*/
title: string;
/**
* key
*/
imageKey?: string;
/**
*
*/
content?: string;
/**
*
*/
confirmText?: string;
/**
*
*/
cancelText?: string;
/**
*
*/
delayText?: string;
} | null;
/**
*
*/
templateInfo: {
/**
* ID
* ${owner}
* ${remindTime}
*/
pendingTemplateId: string;
/**
* ID
* ${owner}
* ${remindTime}
* ${result} textresult对应的
* ${interactTime}
*/
interactedTemplateId: string;
/**
* ID
*/
confirmedTemplateId: string;
/**
* ID
*/
cancelededTemplateId: string;
/**
* ID
*/
delayedTemplateId: string;
} | null;
/**
*
*/
remindTimes: RemindTime[];
/**
*
*/
enabled: boolean;
/**
* yyyy-MM-dd HH:mm
*/
nextRemindTime: string;
/**
*
*/
nextRemindTimeCHS: string;
}
/**
*
*
*/
interface RemindTime {
/**
*
* single: 一次性
* daily: 每天
* weekly: 每周
* monthly: 每月
* yearly: 每年
* workday: 工作日
* holiday: 节假日
*/
frequency:
| "single"
| "daily"
| "weekly"
| "monthly"
| "yearly"
| "workday"
| "holiday";
/**
* 格式为HH:mm single类型时仅作展示用yyyy-MM-dd HH:mm
*/
time: string;
/**
* [1-7]frequency为weekly时有效
*/
daysOfWeek: number[];
/**
* [1-31]frequency为monthly时有效
*/
daysOfMonth: number[];
/**
* frequency为 yearly MM-dd
*/
dayOfYear: string;
}
/**
*
*
*/
interface RemindRecord {
/**
* Id
*/
id: string;
/**
* Id
*/
remindId: string;
/**
* Id
*/
messageId: string;
/**
*
* pending: 待确认
* delay: 已延迟
* confirmed: 已确认
* canceled: 已取消
*/
status: "pending" | "delayed" | "confirmed" | "canceled";
/**
* yyyy-MM-dd HH:mm
*/
remindTime: string;
/**
* yyyy-MM-dd HH:mm
*/
interactTime: string;
/**
* 07:00
*/
result: object;
}
/**
*
*/
interface Header {
/**
* ID
* @example 0f8ab23b60993cf8dd15c8cde4d7b0f5
*/
event_id: string;
/**
* token
* @example tV9djUKSjzVnekV7xTg2Od06NFTcsBnj
*/
token: string;
/**
*
* @example 1693565712117
*/
create_time: string;
/**
*
* @example im.message.receive_v1
*/
event_type: string;
/**
* tenant_key
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
/**
* app_id
* @example cli_a1eff35b43b89063
*/
app_id: string;
}
/**
* ID信息
*/
interface UserIdInfo {
/**
*
* @example ou_032f507d08f9a7f28b042fcd086daef5
*/
open_id: string;
/**
*
* @example on_7111660fddd8302ce47bf1999147c011
*/
union_id: string;
/**
*
* @example zhaoyingbo
*/
user_id: string;
}
/**
* AT的人的信息
*/
interface Mention {
/**
* ID信息
*/
id: UserIdInfo;
/**
*
* @example "@_user_1"
*/
key: string;
/**
*
* @example
*/
name: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
}
/**
*
*/
interface Message {
/**
* ID
* @example oc_433b1cb7a9dbb7ebe70a4e1a59cb8bb1
*/
chat_id: string;
/**
*
* @example group | p2p
*/
chat_type: string;
/**
* JSON字符串文本内容
* @example "{\"text\":\"@_user_1 测试\"}"
*/
content: string;
/**
*
* @example 1693565711996
*/
create_time: string;
/**
*
*/
mentions?: Mention[];
/**
* ID
* @example om_038fc0eceed6224a1abc1cdaa4266405
*/
message_id: string;
/**
*
* @example textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
*/
message_type: string;
}
/**
*
*/
interface Sender {
/**
* id
*/
sender_id: UserIdInfo;
/**
*
* @example user
*/
sender_type: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
}
/**
*
*/
interface Event {
message: Message;
sender: Sender;
}
/**
*
*/
interface LarkMessageEvent {
/**
*
* @example 2.0
*/
schema: string;
/**
*
*/
header: Header;
/**
*
*/
event: Event;
}
/**
* Action信息
*/
interface LarkUserAction {
/**
* open_id
*/
open_id: string;
/**
*
* @example zhaoyingbo
*/
user_id: string;
/**
* ID
* @example om_038fc0eceed6224a1abc1cdaa4266405
*/
open_message_id: string;
/**
* ID
* @example oc_433b1cb7a9dbb7ebe70a4e1a59cb8bb1
*/
open_chat_id: string;
/**
* ID
* @example 2ee61fe50f4f1657
*/
tenant_key: string;
/**
* token
* @example tV9djUKSjzVnekV7xTg2Od06NFTcsBnj
*/
token: string;
/**
*
*/
action: {
/**
*
*/
value: any;
/**
*
* @example picker_datetime
*/
tag: string;
/**
*
* @example 2023-09-03 10:35 +0800
*/
option: string;
/**
*
*/
timezone: string;
};
}
type ReceiveIDType = "open_id" | "user_id" | "union_id" | "email" | "chat_id";
type MsgType = "text" | "post" | "image" | "file" | "audio" | "media" | "sticker" | "interactive" | "share_chat" | "share_user";
interface ServerResponse {
code: number;
data: any;
msg: string;
}

View File

@ -1,42 +1,42 @@
/**
*
* @param {LarkMessageEvent} body
*/
export const getIsEventMsg = (body: LarkMessageEvent) => {
return body?.header?.event_type === "im.message.receive_v1";
};
/**
*
* @param {LarkMessageEvent} body
* @returns
*/
export const getMsgType = (body: LarkMessageEvent) => {
return body?.event?.message?.message_type;
};
/**
* Id
* @param {LarkMessageEvent} body
* @returns
*/
export const getChatId = (body: LarkMessageEvent) => {
return body?.event?.message?.chat_id;
};
/**
* Action消息
* @param {LarkUserAction} body
*/
export const getIsActionMsg = (body: LarkUserAction) => {
return body?.action;
};
/**
* Action类型
* @param {LarkUserAction} body
* @returns {string} Action类型
*/
export const getActionType = (body: LarkUserAction) => {
return body?.action?.tag;
};
/**
*
* @param {LarkMessageEvent} body
*/
export const getIsEventMsg = (body: LarkMessageEvent) => {
return body?.header?.event_type === "im.message.receive_v1";
};
/**
*
* @param {LarkMessageEvent} body
* @returns
*/
export const getMsgType = (body: LarkMessageEvent) => {
return body?.event?.message?.message_type;
};
/**
* Id
* @param {LarkMessageEvent} body
* @returns
*/
export const getChatId = (body: LarkMessageEvent) => {
return body?.event?.message?.chat_id;
};
/**
* Action消息
* @param {LarkUserAction} body
*/
export const getIsActionMsg = (body: LarkUserAction) => {
return body?.action;
};
/**
* Action类型
* @param {LarkUserAction} body
* @returns {string} Action类型
*/
export const getActionType = (body: LarkUserAction) => {
return body?.action?.tag;
};

View File

@ -1,11 +1,11 @@
export const managePb404 = async (dbFunc: Function) => {
try {
return await dbFunc()
} catch (err: any) {
console.log("🚀 ~ manage404 ~ err:", err)
// 没有这个提醒就返回空
if (err?.message === "The requested resource wasn't found.") {
return null
} else throw err;
}
}
export const managePb404 = async (dbFunc: Function) => {
try {
return await dbFunc()
} catch (err: any) {
console.log("🚀 ~ manage404 ~ err:", err)
// 没有这个提醒就返回空
if (err?.message === "The requested resource wasn't found.") {
return null
} else throw err;
}
}

View File

@ -1,115 +1,115 @@
import db from "../db";
/**
*
* @param func fetch
* @returns
*/
const manageFetch = async (func: Function) => {
try {
const res = await func();
const data = (await res.json()) as ServerResponse;
console.log("🚀 ~ manageFetch ~ data:", data);
return data;
} catch (error) {
console.log("🚀 ~ manageFetch ~ error:", error);
return {
code: 1,
data: null,
msg: "sendMsg fetch error",
};
}
};
/**
* header
* @returns header
*/
const getHeaders = async () => {
const tenant_access_token = await db.tenantAccessToken.get();
return {
"Content-Type": "application/json",
Authorization: `Bearer ${tenant_access_token}`,
};
};
/**
*
* @param {ReceiveIDType} receive_id_type id类型 open_id/user_id/union_id/email/chat_id
* @param {string} receive_id IDID类型应与查询参数receive_id_type
* @param {MsgType} msg_type textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
* @param {string} content JSON结构序列化后的字符串msg_type对应不同内容
*/
export const sendMsg = async (
receive_id_type: ReceiveIDType,
receive_id: string,
msg_type: MsgType,
content: string
) => {
const URL = `https://open.f.mioffice.cn/open-apis/im/v1/messages?receive_id_type=${receive_id_type}`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ receive_id, msg_type, content }),
})
);
};
/**
*
* @param {string} message_id id
* @param {string} content JSON结构序列化后的字符串msg_type对应不同内容
*/
export const updateCard = async (message_id: string, content: string) => {
const URL = `https://open.f.mioffice.cn/open-apis/im/v1/messages/${message_id}`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "PATCH",
headers,
body: JSON.stringify({ content }),
})
);
};
/**
*
* @param {string} chat_id ID
* @param {string} open_id ID
* @param {MsgType} msg_type textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
* @param {*} card String
*/
export const sendEphemeralMsg = async (
chat_id: string,
open_id: string,
msg_type: MsgType,
card: any
) => {
const URL = `https://open.f.mioffice.cn/open-apis/ephemeral/v1/send`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ chat_id, open_id, msg_type, card }),
})
);
};
/**
*
* @param message_id id
*/
export const delEphemeralMsg = async (message_id: string) => {
const URL = `https://open.f.mioffice.cn/open-apis/ephemeral/v1/delete`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ message_id }),
})
);
};
import db from "../db";
/**
*
* @param func fetch
* @returns
*/
const manageFetch = async (func: Function) => {
try {
const res = await func();
const data = (await res.json()) as ServerResponse;
console.log("🚀 ~ manageFetch ~ data:", data);
return data;
} catch (error) {
console.log("🚀 ~ manageFetch ~ error:", error);
return {
code: 1,
data: null,
msg: "sendMsg fetch error",
};
}
};
/**
* header
* @returns header
*/
const getHeaders = async () => {
const tenant_access_token = await db.tenantAccessToken.get();
return {
"Content-Type": "application/json",
Authorization: `Bearer ${tenant_access_token}`,
};
};
/**
*
* @param {ReceiveIDType} receive_id_type id类型 open_id/user_id/union_id/email/chat_id
* @param {string} receive_id IDID类型应与查询参数receive_id_type
* @param {MsgType} msg_type textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
* @param {string} content JSON结构序列化后的字符串msg_type对应不同内容
*/
export const sendMsg = async (
receive_id_type: ReceiveIDType,
receive_id: string,
msg_type: MsgType,
content: string
) => {
const URL = `https://open.f.mioffice.cn/open-apis/im/v1/messages?receive_id_type=${receive_id_type}`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ receive_id, msg_type, content }),
})
);
};
/**
*
* @param {string} message_id id
* @param {string} content JSON结构序列化后的字符串msg_type对应不同内容
*/
export const updateCard = async (message_id: string, content: string) => {
const URL = `https://open.f.mioffice.cn/open-apis/im/v1/messages/${message_id}`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "PATCH",
headers,
body: JSON.stringify({ content }),
})
);
};
/**
*
* @param {string} chat_id ID
* @param {string} open_id ID
* @param {MsgType} msg_type textpostimagefileaudiomediastickerinteractiveshare_chatshare_user
* @param {*} card String
*/
export const sendEphemeralMsg = async (
chat_id: string,
open_id: string,
msg_type: MsgType,
card: any
) => {
const URL = `https://open.f.mioffice.cn/open-apis/ephemeral/v1/send`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ chat_id, open_id, msg_type, card }),
})
);
};
/**
*
* @param message_id id
*/
export const delEphemeralMsg = async (message_id: string) => {
const URL = `https://open.f.mioffice.cn/open-apis/ephemeral/v1/delete`;
const headers = await getHeaders();
return await manageFetch(() =>
fetch(URL, {
method: "POST",
headers,
body: JSON.stringify({ message_id }),
})
);
};