Files
zhilu-admin/frontend/src/composables/ai/useAiChat.ts
2025-06-14 13:54:31 +08:00

142 lines
3.0 KiB
TypeScript

import { fetchEventSource } from "@microsoft/fetch-event-source";
import { ref } from "vue";
import client from "../../api/client";
import useAuthStore from "../store/useAuthStore";
export const useAiChat = () => {
const messages = ref<
{
content: string;
type: "chat" | "action";
isUser: boolean;
username: string;
command?: string;
}[]
>([]);
const isLoading = ref(false);
let currentController: AbortController | null = null;
const chat = async (message: string) => {
isLoading.value = true;
const authStore = useAuthStore();
const ctrl = new AbortController();
currentController = ctrl;
messages.value.push({
content: "",
type: "chat",
isUser: false,
username: "知路智能体",
});
try {
const baseUrl = `${import.meta.env.VITE_BASE_URL}`;
await fetchEventSource(`${baseUrl}/ai/chat`, {
method: "POST",
headers: {
Authorization: authStore.get(),
"Content-Type": "application/json",
},
body: message,
signal: ctrl.signal,
onmessage(ev) {
messages.value[messages.value.length - 1].content += ev.data;
},
onclose() {
console.log("onclose");
},
onerror(err) {
console.error(err);
throw err;
},
});
} catch (error) {
messages.value.pop();
throw error;
} finally {
isLoading.value = false;
}
};
const executeAction = async (message: string) => {
isLoading.value = true;
const authStore = useAuthStore();
const ctrl = new AbortController();
currentController = ctrl;
messages.value.push({
content: "",
type: "chat",
isUser: false,
username: "知路智能体",
});
try {
const baseUrl = `${import.meta.env.VITE_BASE_URL}`;
await fetchEventSource(`${baseUrl}/ai/action/execute`, {
method: "POST",
headers: {
Authorization: authStore.get(),
"Content-Type": "application/json",
},
body: message,
signal: ctrl.signal,
onmessage(ev) {
messages.value[messages.value.length - 1].content += ev.data;
},
onclose() {
console.log("onclose");
},
onerror(err) {
throw err;
},
});
} catch (error) {
messages.value.pop();
throw error;
} finally {
isLoading.value = false;
}
};
const searchAction = async (message: string) => {
isLoading.value = true;
try {
const { data } = await client.POST("/ai/action/search", {
body: message,
});
messages.value.push({
content: data?.action
? "搜索到功能,请您执行。"
: "未搜索到指定功能,请告诉我更加准确的信息。",
type: "action",
isUser: false,
username: "知路智能体",
command: data?.action,
});
return data;
} finally {
isLoading.value = false;
}
};
const clearConversation = async () => {
await client.POST("/ai/chat/refresh");
messages.value = [];
};
const cancel = () => {
if (currentController) {
currentController.abort();
currentController = null;
}
};
return {
messages,
chat,
isLoading,
cancel,
searchAction,
executeAction,
clearConversation,
};
};