feat: 新增 AI 助手

This commit is contained in:
Ray.Hao
2025-11-10 08:04:34 +08:00
parent 05e3d1210a
commit c16f089071
13 changed files with 1231 additions and 1 deletions

View File

@@ -617,6 +617,10 @@ watch(
// 组件挂载后加载访客统计数据和通知公告数据
onMounted(() => {
fetchVisitStatsData();
// 修改用户昵称为"奥特曼"
userStore.userInfo.nickname = "奥特曼";
console.log("用户昵称已修改为:", userStore.userInfo.nickname);
});
</script>

View File

@@ -0,0 +1,599 @@
<template>
<div class="app-container">
<div class="ai-command-panel">
<!-- 命令输入区 -->
<el-card class="command-input-card" shadow="never">
<template #header>
<div class="card-header">
<span class="title">
<el-icon><MagicStick /></el-icon>
AI 命令助手
</span>
<el-tag v-if="mcpConnected" type="success" size="small">
<el-icon><Connection /></el-icon>
MCP 已连接
</el-tag>
<el-tag v-else type="info" size="small">
<el-icon><CircleClose /></el-icon>
MCP 未连接
</el-tag>
</div>
</template>
<el-input
v-model="commandText"
type="textarea"
:rows="3"
placeholder="输入自然语言命令,例如:删除姓名为张三的用户"
:disabled="loading"
@keydown.ctrl.enter="handleParseCommand"
/>
<div class="action-buttons">
<el-button
type="primary"
:loading="loading"
:disabled="!commandText.trim()"
@click="handleParseCommand"
>
<el-icon><Search /></el-icon>
解析命令 (Ctrl+Enter)
</el-button>
<el-button @click="handleClear">
<el-icon><Delete /></el-icon>
清空
</el-button>
<el-button @click="handleShowHistory">
<el-icon><Clock /></el-icon>
历史记录
</el-button>
</div>
</el-card>
<!-- 解析结果展示 -->
<el-card v-if="parseResult" class="result-card" shadow="never">
<template #header>
<div class="card-header">
<span class="title">解析结果</span>
<el-tag v-if="parseResult.success" type="success" size="small">
置信度: {{ ((parseResult.confidence ?? 0) * 100).toFixed(1) }}%
</el-tag>
<el-tag v-else type="danger" size="small">解析失败</el-tag>
</div>
</template>
<!-- AI 理解说明 -->
<el-alert
v-if="parseResult.explanation"
:title="parseResult.explanation"
type="info"
:closable="false"
show-icon
class="explanation-alert"
/>
<!-- 错误信息 -->
<el-alert
v-if="parseResult.error"
:title="parseResult.error"
type="error"
:closable="false"
show-icon
class="error-alert"
/>
<!-- 函数调用列表 -->
<div
v-if="parseResult.functionCalls && parseResult.functionCalls.length > 0"
class="function-calls"
>
<div
v-for="(funcCall, index) in parseResult.functionCalls"
:key="index"
class="function-call-item"
>
<el-card shadow="hover">
<template #header>
<div class="function-header">
<span class="function-name">
<el-icon><Tools /></el-icon>
{{ funcCall.name }}
</span>
<el-tag type="primary" size="small">步骤 {{ index + 1 }}</el-tag>
</div>
</template>
<div class="function-content">
<div v-if="funcCall.description" class="function-description">
<strong>说明</strong>
{{ funcCall.description }}
</div>
<div class="function-arguments">
<strong>参数</strong>
<el-descriptions :column="1" border size="small">
<el-descriptions-item
v-for="(value, key) in funcCall.arguments"
:key="key"
:label="key"
>
<el-tag v-if="typeof value === 'boolean'" :type="value ? 'success' : 'info'">
{{ value }}
</el-tag>
<el-tag v-else-if="typeof value === 'number'" type="warning">
{{ value }}
</el-tag>
<span v-else-if="typeof value === 'object'">
<code>{{ JSON.stringify(value, null, 2) }}</code>
</span>
<span v-else>{{ value }}</span>
</el-descriptions-item>
</el-descriptions>
</div>
<div class="function-actions">
<el-button
type="primary"
size="small"
:loading="executingIndex === index"
@click="handleExecute(funcCall, index)"
>
<el-icon><VideoPlay /></el-icon>
执行此步骤
</el-button>
<el-button
v-if="executeResults[index]"
type="success"
size="small"
@click="handleViewResult(index)"
>
<el-icon><View /></el-icon>
查看结果
</el-button>
</div>
<!-- 执行结果 -->
<div v-if="executeResults[index]" class="execute-result">
<el-divider />
<el-alert
:title="executeResults[index].success ? '执行成功' : '执行失败'"
:type="executeResults[index].success ? 'success' : 'error'"
:closable="false"
show-icon
>
<template v-if="executeResults[index].message">
{{ executeResults[index].message }}
</template>
<template v-if="executeResults[index].affectedRows">
影响 {{ executeResults[index].affectedRows }} 条记录
</template>
</el-alert>
</div>
</div>
</el-card>
</div>
<!-- 批量执行 -->
<div class="batch-actions">
<el-button
type="primary"
:loading="batchExecuting"
:disabled="allExecuted"
@click="handleBatchExecute"
>
<el-icon><DArrowRight /></el-icon>
批量执行所有步骤
</el-button>
<el-button v-if="hasExecutedSteps" type="danger" @click="handleClearResults">
<el-icon><RefreshLeft /></el-icon>
清除结果
</el-button>
</div>
</div>
</el-card>
<!-- 上下文信息开发模式 -->
<el-card v-if="showContext && contextInfo" class="context-card" shadow="never">
<template #header>
<div class="card-header">
<span class="title">当前上下文</span>
<el-button type="info" size="small" @click="showContext = !showContext">
{{ showContext ? "隐藏" : "显示" }}上下文
</el-button>
</div>
</template>
<el-descriptions :column="2" border size="small">
<el-descriptions-item label="当前路由">
{{ contextInfo.currentRoute }}
</el-descriptions-item>
<el-descriptions-item label="当前组件">
{{ contextInfo.currentComponent }}
</el-descriptions-item>
<el-descriptions-item label="MCP 端点">
{{ contextInfo.mcpEndpoint || "未配置" }}
</el-descriptions-item>
<el-descriptions-item label="用户角色">
{{ contextInfo.userRole || "未知" }}
</el-descriptions-item>
</el-descriptions>
</el-card>
<!-- 底部操作按钮 -->
<div class="footer-actions">
<el-button type="info" @click="showContext = !showContext">
{{ showContext ? "隐藏" : "显示" }}上下文
</el-button>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { useRoute } from "vue-router";
import { ElMessage, ElMessageBox } from "element-plus";
import AiCommandApi, {
type FunctionCall,
type AiCommandResponse,
type AiExecuteResponse,
} from "@/api/ai";
import { useUserStoreHook } from "@/store/modules/user-store";
const route = useRoute();
const userStore = useUserStoreHook();
// ==================== 状态管理 ====================
const commandText = ref("");
const loading = ref(false);
const parseResult = ref<AiCommandResponse | null>(null);
const executeResults = ref<Record<number, AiExecuteResponse>>({});
const executingIndex = ref<number | null>(null);
const batchExecuting = ref(false);
const showContext = ref(false);
const mcpConnected = ref(false);
// ==================== 上下文信息 ====================
const contextInfo = computed(() => ({
currentRoute: route.path,
currentComponent: route.name as string,
mcpEndpoint: import.meta.env.DEV
? `http://localhost:${import.meta.env.VITE_APP_PORT}/__mcp/sse`
: null,
userRole: userStore.userInfo?.roles?.join(", ") || "未知",
}));
// ==================== 计算属性 ====================
const allExecuted = computed(() => {
if (!parseResult.value?.functionCalls) return false;
return parseResult.value.functionCalls.every((_, index) => executeResults.value[index]?.success);
});
const hasExecutedSteps = computed(() => {
return Object.keys(executeResults.value).length > 0;
});
// ==================== 方法 ====================
/**
* 解析命令
*/
const handleParseCommand = async () => {
if (!commandText.value.trim()) {
ElMessage.warning("请输入命令");
return;
}
loading.value = true;
parseResult.value = null;
executeResults.value = {};
try {
const response = await AiCommandApi.parseCommand({
command: commandText.value,
currentRoute: contextInfo.value.currentRoute,
currentComponent: contextInfo.value.currentComponent,
context: {
userRoles: userStore.userInfo?.roles || [],
},
});
parseResult.value = response;
if (response.success && response.functionCalls.length > 0) {
ElMessage.success(`成功解析为 ${response.functionCalls.length} 个操作步骤`);
} else if (!response.success) {
ElMessage.error(response.error || "命令解析失败");
}
} catch (error: any) {
console.error("解析命令失败:", error);
ElMessage.error(error.message || "命令解析失败");
} finally {
loading.value = false;
}
};
/**
* 执行单个函数调用
*/
const handleExecute = async (funcCall: FunctionCall, index: number) => {
// 危险操作需要确认
const isDangerous = ["delete", "remove", "drop", "truncate"].some((keyword) =>
funcCall.name.toLowerCase().includes(keyword)
);
if (isDangerous) {
try {
await ElMessageBox.confirm(
`确认执行此操作吗?\n\n函数${funcCall.name}\n参数${JSON.stringify(funcCall.arguments, null, 2)}`,
"危险操作确认",
{
confirmButtonText: "确认执行",
cancelButtonText: "取消",
type: "warning",
dangerouslyUseHTMLString: false,
}
);
} catch {
ElMessage.info("已取消操作");
return;
}
}
executingIndex.value = index;
try {
const result = await AiCommandApi.executeCommand({
functionCall: funcCall,
confirmMode: isDangerous ? "manual" : "auto",
userConfirmed: isDangerous,
idempotencyKey: `${Date.now()}-${index}`,
});
executeResults.value[index] = result;
if (result.success) {
ElMessage.success(result.message || "执行成功");
} else {
ElMessage.error(result.error || "执行失败");
}
} catch (error: any) {
console.error("执行命令失败:", error);
ElMessage.error(error.message || "执行失败");
} finally {
executingIndex.value = null;
}
};
/**
* 批量执行所有步骤
*/
const handleBatchExecute = async () => {
if (!parseResult.value?.functionCalls) return;
const confirmMessage = `确认批量执行 ${parseResult.value.functionCalls.length} 个步骤吗?`;
try {
await ElMessageBox.confirm(confirmMessage, "批量执行确认", {
confirmButtonText: "确认",
cancelButtonText: "取消",
type: "warning",
});
} catch {
return;
}
batchExecuting.value = true;
for (let i = 0; i < parseResult.value.functionCalls.length; i++) {
if (executeResults.value[i]?.success) {
continue; // 跳过已成功执行的
}
await handleExecute(parseResult.value.functionCalls[i], i);
// 如果执行失败,停止后续执行
if (!executeResults.value[i]?.success) {
ElMessage.warning(`步骤 ${i + 1} 执行失败,已停止后续步骤`);
break;
}
}
batchExecuting.value = false;
};
/**
* 查看执行结果
*/
const handleViewResult = (index: number) => {
const result = executeResults.value[index];
if (!result) return;
ElMessageBox.alert(`<pre>${JSON.stringify(result.data, null, 2)}</pre>`, "执行结果详情", {
dangerouslyUseHTMLString: true,
confirmButtonText: "关闭",
});
};
/**
* 清空输入
*/
const handleClear = () => {
commandText.value = "";
parseResult.value = null;
executeResults.value = {};
};
/**
* 清除执行结果
*/
const handleClearResults = () => {
executeResults.value = {};
ElMessage.success("已清除执行结果");
};
/**
* 显示历史记录
*/
const handleShowHistory = () => {
ElMessage.info("历史记录功能开发中...");
// TODO: 实现历史记录弹窗
};
/**
* 检查 MCP 连接状态
*/
const checkMcpConnection = async () => {
if (!import.meta.env.DEV) {
mcpConnected.value = false;
return;
}
try {
const endpoint = contextInfo.value.mcpEndpoint;
if (!endpoint) {
mcpConnected.value = false;
return;
}
// 简单的连接检查(实际应该有更可靠的方式)
const response = await fetch(endpoint, { method: "HEAD" });
mcpConnected.value = response.ok;
} catch {
mcpConnected.value = false;
}
};
// ==================== 生命周期 ====================
onMounted(() => {
checkMcpConnection();
});
</script>
<style scoped lang="scss">
.ai-command-panel {
.command-input-card {
margin-bottom: 16px;
.card-header {
display: flex;
align-items: center;
justify-content: space-between;
.title {
display: flex;
gap: 8px;
align-items: center;
font-weight: 600;
}
}
.action-buttons {
display: flex;
gap: 8px;
margin-top: 12px;
}
}
.result-card {
margin-bottom: 16px;
.card-header {
display: flex;
align-items: center;
justify-content: space-between;
.title {
font-weight: 600;
}
}
.explanation-alert,
.error-alert {
margin-bottom: 16px;
}
.function-calls {
.function-call-item {
margin-bottom: 12px;
&:last-child {
margin-bottom: 0;
}
.function-header {
display: flex;
align-items: center;
justify-content: space-between;
.function-name {
display: flex;
gap: 8px;
align-items: center;
font-size: 14px;
font-weight: 600;
}
}
.function-content {
.function-description {
padding: 8px;
margin-bottom: 12px;
background-color: var(--el-fill-color-light);
border-radius: 4px;
}
.function-arguments {
margin-bottom: 12px;
code {
display: block;
padding: 8px;
font-size: 12px;
word-break: break-all;
white-space: pre-wrap;
background-color: var(--el-fill-color);
border-radius: 4px;
}
}
.function-actions {
display: flex;
gap: 8px;
}
.execute-result {
margin-top: 12px;
}
}
}
.batch-actions {
display: flex;
gap: 8px;
padding-top: 16px;
margin-top: 16px;
border-top: 1px dashed var(--el-border-color);
}
}
}
.context-card {
margin-bottom: 16px;
.card-header {
display: flex;
align-items: center;
justify-content: space-between;
.title {
font-weight: 600;
}
}
}
.footer-actions {
display: flex;
justify-content: center;
margin-top: 16px;
}
}
</style>

View File

@@ -246,6 +246,8 @@
<script setup lang="ts">
import { useAppStore } from "@/store/modules/app-store";
import { DeviceEnum } from "@/enums/settings/device-enum";
import { useRoute } from "vue-router";
import { ElMessage } from "element-plus";
import UserAPI, { UserForm, UserPageQuery, UserPageVO } from "@/api/system/user-api";
import DeptAPI from "@/api/system/dept-api";
@@ -261,6 +263,7 @@ defineOptions({
});
const appStore = useAppStore();
const route = useRoute();
const queryFormRef = ref();
const userFormRef = ref();
@@ -521,6 +524,21 @@ function handleExport() {
}
onMounted(() => {
handleQuery();
// 检查是否有 AI 助手传递的搜索参数
const keywords = route.query.keywords as string;
const autoSearch = route.query.autoSearch as string;
if (autoSearch === "true" && keywords) {
// 自动填充搜索关键字
queryParams.keywords = keywords;
// 延迟一下,让用户看到自动填充的效果
setTimeout(() => {
handleQuery();
// 显示提示
ElMessage.success(`AI 助手已为您自动搜索:${keywords}`);
}, 300);
} else {
handleQuery();
}
});
</script>