diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index 33f42ec..daa8c66 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -30,7 +30,8 @@ "Prisma.prisma", "humao.rest-client", "GitHub.copilot", - "GitHub.copilot-chat" + "GitHub.copilot-chat", + "oven.bun-vscode" ] } }, diff --git a/.env.dev b/.env.dev deleted file mode 100644 index dcd53a1..0000000 --- a/.env.dev +++ /dev/null @@ -1,4 +0,0 @@ -NODE_ENV=dev -http_proxy=http://proxy.pt.xiaomi.com:80 -https_proxy=http://proxy.pt.xiaomi.com:80 -no_proxy=localhost,.xiaomiwh.cn,.mioffice.cn,.xiaomi.com,.imoaix.cn,.deepseek.com,.gpt.ge,.xiaomi.srv \ No newline at end of file diff --git a/.env.prod b/.env.prod deleted file mode 100644 index 58c0f39..0000000 --- a/.env.prod +++ /dev/null @@ -1,4 +0,0 @@ -NODE_ENV=production -http_proxy=http://proxy.pt.xiaomi.com:80 -https_proxy=http://proxy.pt.xiaomi.com:80 -no_proxy=localhost,.xiaomiwh.cn,.mioffice.cn,.xiaomi.com,.imoaix.cn,.deepseek.com,.gpt.ge,.xiaomi.srv \ No newline at end of file diff --git a/.gitea/workflows/miflow.yml b/.gitea/workflows/miflow.yml.bak similarity index 100% rename from .gitea/workflows/miflow.yml rename to .gitea/workflows/miflow.yml.bak diff --git a/.gitignore b/.gitignore index 96c022e..38ad061 100644 --- a/.gitignore +++ b/.gitignore @@ -56,3 +56,5 @@ profile-* profile* *clinic* *flamegraph* + +.env \ No newline at end of file diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml new file mode 100644 index 0000000..7fe72a1 --- /dev/null +++ b/.gitlab-ci.yml @@ -0,0 +1,38 @@ +variables: + REGISTRY: micr.cloud.mioffice.cn + IMAGE_NAME: micr.cloud.mioffice.cn/egg/egg_server + MATRIX_PROJECT: 462609 + MATRIX_SPACE: egg-server-prev + MATRIX_AK: CAKBCYGR0BI2BGMVUF8 + +stages: + - build + - deploy + +default: + tags: + - fe-bj + +build: + stage: build + only: + - master + image: docker:latest + services: + - docker:dind + script: + - cp $ENV .env + - docker login -u $MATRIX_AK -p $MATRIX_SK $REGISTRY + - docker build -t $IMAGE_NAME:$CI_COMMIT_SHA -f ./docker/deploy/Dockerfile . + - docker push $IMAGE_NAME:$CI_COMMIT_SHA + - docker logout $REGISTRY + - docker rmi $IMAGE_NAME:$CI_COMMIT_SHA + +deploy: + stage: deploy + only: + - master + image: cr.d.xiaomi.net/bigdata-fe/matrix-cli + script: + - matrix-cli config set --access-key $MATRIX_AK --secret-key $MATRIX_SK + - matrix-cli deploy --project $MATRIX_PROJECT --deploy-space $MATRIX_SPACE --reason "$CI_COMMIT_MESSAGE" --deploy-percentage 1 --tag $CI_COMMIT_SHA diff --git a/.vscode/settings.json b/.vscode/settings.json index 7b56f55..565a391 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -25,6 +25,7 @@ "mindnote", "openai", "openchat", + "oxlint", "pocketbase", "PWTHWP", "qwen", diff --git a/bun.lockb b/bun.lockb index 027941d..c83e5dd 100755 Binary files a/bun.lockb and b/bun.lockb differ diff --git a/cache.ts b/cache.ts new file mode 100644 index 0000000..db0fd97 --- /dev/null +++ b/cache.ts @@ -0,0 +1,3 @@ +import { parseJsonString } from "@egg/hooks" + +console.log(parseJsonString(Bun.env.LARK_APP_MAP, []), []) diff --git a/Dockerfile b/docker/deploy/Dockerfile similarity index 100% rename from Dockerfile rename to docker/deploy/Dockerfile diff --git a/eslint.config.js b/eslint.config.js index d871960..7540c69 100644 --- a/eslint.config.js +++ b/eslint.config.js @@ -1,5 +1,6 @@ import pluginJs from "@eslint/js" import simpleImportSort from "eslint-plugin-simple-import-sort" +import unusedImports from "eslint-plugin-unused-imports" import globals from "globals" import tseslint from "typescript-eslint" @@ -11,12 +12,24 @@ export default [ { plugins: { "simple-import-sort": simpleImportSort, + "unused-imports": unusedImports, }, rules: { "@typescript-eslint/no-explicit-any": "off", "@typescript-eslint/no-namespace": "off", "simple-import-sort/imports": "error", "simple-import-sort/exports": "error", + "no-unused-vars": "off", // Turn off base rule + "unused-imports/no-unused-imports": "error", // Enable plugin rule + "unused-imports/no-unused-vars": [ + "warn", + { + vars: "all", + varsIgnorePattern: "^_", + args: "after-used", + argsIgnorePattern: "^_", + }, + ], }, }, ] diff --git a/package.json b/package.json index 573ee3c..535c27e 100644 --- a/package.json +++ b/package.json @@ -3,50 +3,51 @@ "module": "index.ts", "type": "module", "scripts": { - "start": "dotenvx run -f .env.prod -- bun run index.ts", - "dev": "dotenvx run -f .env.dev -- bun run index.ts --watch", - "lint": "eslint --fix .", + "start": "bun run index.ts", + "dev": "bun run index.ts --watch", + "lint": "oxlint --fix .", "prepare": "husky", "prettier": "prettier --write ." }, "lint-staged": { "*.{js,jsx,ts,tsx}": [ - "eslint --fix", + "oxlint --fix", "prettier --write", "git add" ] }, "devDependencies": { - "@commitlint/cli": "^19.5.0", - "@commitlint/config-conventional": "^19.5.0", - "@eslint/js": "^9.13.0", + "@commitlint/cli": "^19.6.0", + "@commitlint/config-conventional": "^19.6.0", + "@eslint/js": "^9.15.0", "@types/node-schedule": "^2.1.7", "@types/uuid": "^10.0.0", - "bun-types": "^1.1.33", - "eslint": "^9.13.0", + "bun-types": "^1.1.36", + "eslint": "^9.15.0", "eslint-plugin-simple-import-sort": "^12.1.1", - "husky": "^9.1.6", + "eslint-plugin-unused-imports": "^4.1.4", + "husky": "^9.1.7", "lint-staged": "^15.2.10", + "oxlint": "^0.13.1", "prettier": "^3.3.3", - "typescript-eslint": "^8.12.1" + "typescript-eslint": "^8.15.0" }, "peerDependencies": { "typescript": "^5.5.4" }, "dependencies": { - "@dotenvx/dotenvx": "^1.21.0", "@egg/hooks": "^1.2.0", "@egg/lark-msg-tool": "^1.14.0", "@egg/logger": "^1.4.4", - "@egg/net-tool": "^1.9.2", + "@egg/net-tool": "^1.14.2", "@egg/path-tool": "^1.4.1", - "@langchain/core": "^0.3.15", - "@langchain/openai": "^0.3.11", + "@langchain/core": "^0.3.18", + "@langchain/openai": "^0.3.14", "joi": "^17.13.3", - "langfuse-langchain": "^3.28.0", + "langfuse-langchain": "^3.30.3", "node-schedule": "^2.1.1", "p-limit": "^6.1.0", "pocketbase": "^0.21.5", "uuid": "^10.0.0" } -} +} \ No newline at end of file diff --git a/routes/message/index.ts b/routes/message/index.ts index cf3f4ce..2906c8a 100644 --- a/routes/message/index.ts +++ b/routes/message/index.ts @@ -1,4 +1,5 @@ import { stringifyJson } from "@egg/hooks" +import { LarkService } from "@egg/net-tool" import db from "../../db" import { Context, DB, LarkServer, MsgProxy } from "../../types" @@ -40,7 +41,7 @@ const validateMessageReq = ({ export const manageMessageReq = async ( ctx: Context.Data ): Promise => { - const { body: rawBody, genResp, larkService } = ctx + const { body: rawBody, genResp, appMap, requestId } = ctx const body = rawBody as MsgProxy.Body // 校验参数 @@ -87,6 +88,20 @@ export const manageMessageReq = async ( return genResp.notFound(error) } + // 获取 app info + const appInfo = appMap[appName] + if (!appInfo) { + const error = "app not found" + db.log.create(LOG_COLLECTION, { ...baseLog, error }) + return genResp.notFound(error) + } + + const larkService = new LarkService({ + appId: appInfo.app_id, + appSecret: appInfo.app_secret, + requestId, + }) + // 如果有 group_id,则发送给所有 group_id 中的人 if (body.group_id) { // 获取所有接收者 @@ -103,14 +118,8 @@ export const manageMessageReq = async ( const makeSendFunc = (receive_id_type: LarkServer.ReceiveIDType) => { return (receive_id: string) => { sendList.push( - larkService - .child(appName) - .message.send( - receive_id_type, - receive_id, - body.msg_type, - finalContent - ) + larkService.message + .send(receive_id_type, receive_id, body.msg_type, finalContent) .then((res) => { sendRes[receive_id_type][receive_id] = res }) @@ -130,14 +139,8 @@ export const manageMessageReq = async ( if (body.receive_id && body.receive_id_type) { body.receive_id.split(",").forEach((receive_id) => { sendList.push( - larkService - .child(appName) - .message.send( - body.receive_id_type, - receive_id, - body.msg_type, - finalContent - ) + larkService.message + .send(body.receive_id_type, receive_id, body.msg_type, finalContent) .then((res) => { sendRes[body.receive_id_type][receive_id] = res }) diff --git a/routes/microApp/index.ts b/routes/microApp/index.ts index 81a81f9..2d8d25a 100644 --- a/routes/microApp/index.ts +++ b/routes/microApp/index.ts @@ -1,3 +1,5 @@ +import { LarkService } from "@egg/net-tool" + import { Context } from "../../types" /** @@ -6,19 +8,34 @@ import { Context } from "../../types" * @returns */ const manageLogin = async (ctx: Context.Data) => { - const { req, larkService, genResp, logger } = ctx + const { req, genResp, logger, appMap, requestId } = ctx logger.info("micro app login") const url = new URL(req.url) const code = url.searchParams.get("code") - const appName = url.searchParams.get("app_name") || undefined + const appName = url.searchParams.get("app_name") if (!code) { return genResp.badRequest("code not found") } + if (!appName) { + return genResp.badRequest("app_name not found") + } + // 获取 app info + const appInfo = appMap[appName] + if (!appInfo) { + return genResp.badRequest("app not found") + } + + const larkService = new LarkService({ + appId: appInfo.app_id, + appSecret: appInfo.app_secret, + requestId, + }) + const { code: resCode, data, message, - } = await larkService.child(appName).user.code2Login(code) + } = await larkService.user.code2Login(code) logger.debug(`get user session: ${JSON.stringify(data)}`) @@ -35,7 +52,7 @@ const manageLogin = async (ctx: Context.Data) => { * @returns */ const manageBatchUser = async (ctx: Context.Data) => { - const { body, genResp, larkService, logger } = ctx + const { body, genResp, logger, appMap, requestId } = ctx logger.info("batch get user info") if (!body) return genResp.badRequest("req body is empty") const { user_ids, user_id_type, app_name } = body @@ -45,10 +62,25 @@ const manageBatchUser = async (ctx: Context.Data) => { if (!user_id_type) { return genResp.badRequest("user_id_type not found") } + if (!app_name) { + return genResp.badRequest("app_name not found") + } + // 获取 app info + const appInfo = appMap[app_name] + if (!appInfo) { + return genResp.badRequest("app not found") + } - const { code, data, message } = await larkService - .child(app_name) - .user.batchGet(user_ids, user_id_type) + const larkService = new LarkService({ + appId: appInfo.app_id, + appSecret: appInfo.app_secret, + requestId, + }) + + const { code, data, message } = await larkService.user.batchGet( + user_ids, + user_id_type + ) logger.debug(`batch get user info: ${JSON.stringify(data)}`) diff --git a/routes/sheet/index.ts b/routes/sheet/index.ts index cd97898..f118aab 100644 --- a/routes/sheet/index.ts +++ b/routes/sheet/index.ts @@ -1,3 +1,4 @@ +import { LarkService } from "@egg/net-tool" import Joi from "joi" import db from "../../db" @@ -10,11 +11,10 @@ import insertSheet from "./insert" * @param {Context.Data} ctx - 上下文数据,包含请求体和响应生成器 * @returns {Promise} 如果校验失败,返回响应对象;否则返回 false */ -const validateSheetReq = async ( - ctx: Context.Data -): Promise => { - const { body, genResp } = ctx - +const validateSheetReq = async ({ + body, + genResp, +}: Context.Data): Promise => { // 定义基础的Schema let schema = Joi.object({ api_key: Joi.string() @@ -64,7 +64,7 @@ const validateSheetReq = async ( * @returns {Promise} 返回响应对象 */ export const manageSheetReq = async (ctx: Context.Data): Promise => { - const { body: rawBody, genResp } = ctx + const { body: rawBody, genResp, appMap, requestId } = ctx const body = rawBody as SheetProxy.InsertData // 校验参数 @@ -83,8 +83,21 @@ export const manageSheetReq = async (ctx: Context.Data): Promise => { return genResp.notFound("app name not found") } + // 获取APP信息 + const appInfo = appMap[appName] + if (!appInfo) { + return genResp.notFound("app not found") + } + + // 组织新的LarkService + ctx.larkService = new LarkService({ + appId: appInfo.app_id, + appSecret: appInfo.app_secret, + requestId, + }) + // 根据请求类型处理 - if (body.type === "insert") return await insertSheet(ctx, appName) + if (body.type === "insert") return await insertSheet(ctx) // 默认返回成功响应 return genResp.ok() diff --git a/routes/sheet/insert.ts b/routes/sheet/insert.ts index 981299d..426876a 100644 --- a/routes/sheet/insert.ts +++ b/routes/sheet/insert.ts @@ -7,13 +7,15 @@ import { SheetProxy } from "../../types/sheetProxy" * @param {string} appName - 应用名称 * @returns {Promise} 返回响应对象 */ -const insertSheet = async (ctx: Context.Data, appName: string) => { +const insertSheet = async (ctx: Context.Data) => { const { genResp, larkService } = ctx const body = ctx.body as SheetProxy.InsertData - const insertRes = await larkService - .child(appName) - .sheet.insertRows(body.sheet_token, body.range, body.values) + const insertRes = await larkService.sheet.insertRows( + body.sheet_token, + body.range, + body.values + ) if (insertRes?.code !== 0) { return genResp.serverError(insertRes?.message) diff --git a/schedule/accessToken.ts b/schedule/accessToken.ts index 25d2891..d311ab7 100644 --- a/schedule/accessToken.ts +++ b/schedule/accessToken.ts @@ -1,24 +1,24 @@ import logger from "@egg/logger" +import { LarkService } from "@egg/net-tool" import pLimit from "p-limit" import db from "../db" -import { LarkService } from "../services" export const resetAccessToken = async () => { try { const appList = await db.appInfo.getFullList() const limit = pLimit(3) - const service = new LarkService("", "schedule") const promiseList = appList.map((app) => limit(() => - service.auth.getAk(app.app_id, app.app_secret).then((res) => { - if (res.code !== 0) return - return db.tenantAccessToken.update( - app.id, - app.name, - res.tenant_access_token - ) - }) + new LarkService({ + appId: app.app_id, + appSecret: app.app_secret, + requestId: "schedule", + }).auth + .getAppAuth() + .then((res) => { + return db.tenantAccessToken.update(app.id, app.name, res) + }) ) ) await Promise.allSettled(promiseList) diff --git a/schedule/index.ts b/schedule/index.ts index 4cce843..88520d4 100644 --- a/schedule/index.ts +++ b/schedule/index.ts @@ -1,10 +1,6 @@ -import schedule from "node-schedule" - -import { resetAccessToken } from "./accessToken" - export const initSchedule = async () => { - // 定时任务,每15分钟刷新一次token - schedule.scheduleJob("*/15 * * * *", resetAccessToken) - // 立即执行一次 - resetAccessToken() + // // 定时任务,每15分钟刷新一次token + // schedule.scheduleJob("*/15 * * * *", resetAccessToken) + // // 立即执行一次 + // resetAccessToken() } diff --git a/services/index.ts b/services/index.ts index 0b646ab..24024cc 100644 --- a/services/index.ts +++ b/services/index.ts @@ -1,4 +1,3 @@ import AttachService from "./attach" -import LarkService from "./lark" -export { AttachService, LarkService } +export { AttachService } diff --git a/test/archive/copyFile.ts b/test/archive/copyFile.ts index c015629..df8b38a 100644 --- a/test/archive/copyFile.ts +++ b/test/archive/copyFile.ts @@ -1,6 +1,10 @@ -import { LarkService } from "../../services" +import { LarkService } from "@egg/net-tool" -const service = new LarkService("egg", "") +const service = new LarkService({ + appId: "cli_9f4b3b1b1b", + appSecret: "bask4IvKT61xCvTUxIkcMaWoiYf", + requestId: "123", +}) const { data: { diff --git a/test/archive/createFile.ts b/test/archive/createFile.ts index 5a307ef..4cc91e5 100644 --- a/test/archive/createFile.ts +++ b/test/archive/createFile.ts @@ -1,8 +1,12 @@ -import LarkDriveService from "../../services/lark/drive" +import { LarkService } from "@egg/net-tool" -const service = new LarkDriveService("egg", "") +const service = new LarkService({ + appId: "cli_9f1e6b4e", + appSecret: "7b2f2f3a9b2c8e8a2f1a2b2b2b2b2b2b", + requestId: "schedule", +}) -const res = await service.createFile( +const res = await service.drive.createFile( "D6ETfzaU9lN08adVDz3kjLey4Bx", "xxx 项目 KV管理器", "bitable" diff --git a/test/archive/getInnerList.ts b/test/archive/getInnerList.ts deleted file mode 100644 index 7b011ba..0000000 --- a/test/archive/getInnerList.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { LarkService } from "../../services" - -const service = new LarkService("egg", "") - -const res = await service.chat.getInnerList() - -console.log(JSON.stringify(res, null, 2)) diff --git a/test/archive/sheet.record.get.test.ts b/test/archive/sheet.record.get.test.ts deleted file mode 100644 index 46b13d3..0000000 --- a/test/archive/sheet.record.get.test.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { test } from "bun:test" - -import LarkSheetService from "../../services/lark/sheet" - -const service = new LarkSheetService("egg", "") - -test("getRecords", async () => { - const res = await service.getRecords( - "bask4BA989TBbnu5R7Onmdh1csb", - "tblabYZk3AYtGLSe" - ) - console.log(res) -}) diff --git a/types/context.ts b/types/context.ts index c6ecaed..a823225 100644 --- a/types/context.ts +++ b/types/context.ts @@ -1,14 +1,19 @@ import { LarkBody, LarkCard } from "@egg/lark-msg-tool" -import { NetTool } from "@egg/net-tool" +import { LarkService, NetTool } from "@egg/net-tool" import { PathCheckTool } from "@egg/path-tool" import { Logger } from "winston" import cardMap from "../constant/card" import functionMap from "../constant/function" import tempMap from "../constant/template" -import { AttachService, LarkService } from "../services" +import { AttachService } from "../services" export namespace Context { + export interface APP_INFO { + app_id: string + app_secret: string + app_name: string + } export interface Data { req: Request requestId: string @@ -23,5 +28,7 @@ export namespace Context { path: PathCheckTool searchParams: URLSearchParams app: "michat" | "egg" | string + appInfo: APP_INFO + appMap: Record } } diff --git a/utils/genContext.ts b/utils/genContext.ts index 4a068fd..4f9adbd 100644 --- a/utils/genContext.ts +++ b/utils/genContext.ts @@ -1,15 +1,22 @@ +import { parseJsonString } from "@egg/hooks" import { LarkBody, LarkCard } from "@egg/lark-msg-tool" import loggerIns from "@egg/logger" -import { NetTool } from "@egg/net-tool" +import { LarkService, NetTool } from "@egg/net-tool" import { PathCheckTool } from "@egg/path-tool" import { v4 as uuid } from "uuid" import cardMap from "../constant/card" import functionMap from "../constant/function" import tempMap from "../constant/template" -import { AttachService, LarkService } from "../services" +import { AttachService } from "../services" import { Context } from "../types" +// 获取所有应用信息 +const appMap = parseJsonString(process.env.LARK_APP_MAP, {}) as Record< + string, + Context.APP_INFO +> + /** * 获取之前的requestId。 * @@ -41,10 +48,15 @@ const genContext = async (req: Request) => { const larkBody = new LarkBody(body) const searchParams = new URL(req.url).searchParams const app = searchParams.get("app") || "egg" + const appInfo = appMap[app] const requestId = getPreRequestId(larkBody) || uuid() const logger = loggerIns.child({ requestId }) const genResp = new NetTool({ requestId }) - const larkService = new LarkService(app, requestId) + const larkService = new LarkService({ + appId: appInfo.app_id, + appSecret: appInfo.app_secret, + requestId, + }) const attachService = new AttachService({ requestId }) const path = new PathCheckTool(req.url) const larkCard = new LarkCard( @@ -70,6 +82,8 @@ const genContext = async (req: Request) => { attachService, searchParams, app, + appInfo, + appMap, } as Context.Data } diff --git a/utils/llm/base.ts b/utils/llm/base.ts new file mode 100644 index 0000000..d64de5f --- /dev/null +++ b/utils/llm/base.ts @@ -0,0 +1,38 @@ +import { ChatOpenAI } from "@langchain/openai" +import CallbackHandler, { Langfuse } from "langfuse-langchain" + +/** + * 获取模型 + * @param temperature 温度 + * @returns + */ +export const getModel = (temperature = 0) => { + return new ChatOpenAI( + { + temperature, + model: Bun.env.LLM_MODEL, + apiKey: Bun.env.LLM_API_KEY, + }, + { + baseURL: Bun.env.LLM_BASE_URL, + } + ) +} + +/** + * 获取Langfuse + * @returns + */ +export const getLangfuse = async (name: string, requestId: string) => { + const langfuseParams = { + publicKey: Bun.env.LANGFUSE_PK, + secretKey: Bun.env.LANGFUSE_SK, + baseUrl: Bun.env.LANGFUSE_BASE_URL, + sessionId: requestId, + name, + } + return { + langfuseHandler: new CallbackHandler(langfuseParams), + langfuse: new Langfuse(langfuseParams), + } +} diff --git a/utils/llm.ts b/utils/llm/index.ts similarity index 61% rename from utils/llm.ts rename to utils/llm/index.ts index be9a62d..86441c0 100644 --- a/utils/llm.ts +++ b/utils/llm/index.ts @@ -1,66 +1,7 @@ import { PromptTemplate } from "@langchain/core/prompts" -import { ChatOpenAI } from "@langchain/openai" -import { CallbackHandler, Langfuse } from "langfuse-langchain" import { z } from "zod" -import db from "../db" - -/** - * 获取Langfuse - * @returns - */ -const getLangfuse = async (name: string, requestId: string) => { - const langfuseParams = { - publicKey: await db.appConfig.getLangfusePk(), - secretKey: await db.appConfig.getLangfuseSk(), - baseUrl: "http://langfuse.c5-cloudml.xiaomi.srv", - sessionId: requestId, - name, - } - return { - langfuseHandler: new CallbackHandler(langfuseParams), - langfuse: new Langfuse(langfuseParams), - } -} - -const modelMap = { - "deepseek-chat": { - model: "deepseek-chat", - apiKey: "xx", - baseURL: "http://10.38.214.162:8003/v1", - }, - "qwen2-72b-instruct-int4": { - model: "qwen2-72b-instruct-int4", - apiKey: "xx", - baseURL: "http://10.38.214.206:8000/v1", - }, - "gpt-4o": { - model: "gpt-4o", - apiKey: "sk-EhbBTR0QjhH22iLr9aCb04D2B0F44f88A07c2924Eb54CfA4", - baseURL: "https://api.gpt.ge/v1", - }, - "qwen-72b-instruct-int4/v1": { - model: "qwen-72b-instruct-int4/v1", - apiKey: "xx", - baseURL: - "http://ms-13871-qwen-model-128k-9-1012195754.kscn-tj5-prod2-cloudml.xiaomi.srv/v1", - }, -} - -/** - * 获取模型 - * @param modelName 模型名称 - * @param temperature 温度 - */ -const getModel = async (modelName: keyof typeof modelMap, temperature = 0) => { - const { model, apiKey, baseURL } = modelMap[modelName] - return new ChatOpenAI( - { temperature, model, apiKey }, - { - baseURL, - } - ) -} +import { getLangfuse, getModel } from "./base" const groupAgentConfig = z.object({ chatId: z.string().describe("群聊ID"), @@ -89,7 +30,7 @@ const parseGroupAgentQuery = async ( "parseGroupAgentQuery", requestId ) - const model = await getModel("qwen-72b-instruct-int4/v1") + const model = await getModel() const structuredLlm = model.withStructuredOutput(groupAgentConfig, { name: "groupAgent", }) @@ -140,7 +81,6 @@ const invoke = async ( ) => { const { langfuse, langfuseHandler } = await getLangfuse("invoke", requestId) const prompt = await langfuse.getPrompt(promptName) - const config = prompt.config as { modelName: keyof typeof modelMap } const langchainTextPrompt = PromptTemplate.fromTemplate( prompt.getLangchainPrompt() @@ -148,9 +88,7 @@ const invoke = async ( metadata: { langfusePrompt: prompt }, }) - const chain = langchainTextPrompt.pipe( - await getModel(config.modelName, temperature) - ) + const chain = langchainTextPrompt.pipe(await getModel(temperature)) const { content } = await chain.invoke(variables, { callbacks: [langfuseHandler],