index.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. <template>
  2. <div class="app-container">
  3. <div class="pagination-container" v-loading.fullscreen.lock="loading">
  4. <div class="tasks-container">
  5. <div class="tasks">
  6. <div
  7. class="task-item"
  8. v-for="task in questionTasks"
  9. :key="task.task_id"
  10. :class="{ 'task-item-selected': selectedTaskId === task.task_id }"
  11. @click="handleTaskClick(task.task_id)"
  12. >
  13. <div class="task-question">{{ task.question }}</div>
  14. </div>
  15. </div>
  16. <div class="pagination">
  17. <el-pagination
  18. @current-change="handleCurrentChange"
  19. :current-page="pageNumber"
  20. :page-size="pageSize"
  21. :total="total"
  22. layout="prev, pager, next"
  23. />
  24. </div>
  25. </div>
  26. <div class="task-detail-container">
  27. <el-descriptions
  28. class="margin-top"
  29. title="任务详情"
  30. :column="3"
  31. border
  32. v-if="selectedTask"
  33. >
  34. <el-descriptions-item label="任务ID">
  35. {{ selectedTask?.task_id }}
  36. </el-descriptions-item>
  37. <el-descriptions-item label="问题">
  38. {{ selectedTask?.question }}
  39. </el-descriptions-item>
  40. <el-descriptions-item label="任务状态">{{ selectedTask?.statusText }}</el-descriptions-item>
  41. <el-descriptions-item label="知识类别">
  42. {{ KnowledgeType[selectedTask?.knowledgeType] || '' }}
  43. </el-descriptions-item>
  44. <el-descriptions-item label="内容知识子类别">
  45. {{ QueryTypeEnum[selectedTask?.query_type] || '-' }}
  46. </el-descriptions-item>
  47. <el-descriptions-item label="是否需要存储">
  48. {{ selectedTask?.need_store === NeedStoreEnum.需要存储 ? '是' : '否' }}
  49. </el-descriptions-item>
  50. </el-descriptions>
  51. <div class="querys-header">Query词条</div>
  52. <el-collapse accordion @change="handleCollapseChange">
  53. <el-collapse-item
  54. v-for="query in selectedQueries"
  55. :key="query.query"
  56. :title="query.query"
  57. :name="query.query"
  58. class="query-title"
  59. >
  60. <div v-loading="query.loading" class="query-data-container">
  61. <div v-if="query.data" class="query-data-content">
  62. {{ query.data }}
  63. </div>
  64. <div v-else-if="!query.loading" class="query-data-empty">
  65. 暂无数据
  66. </div>
  67. </div>
  68. </el-collapse-item>
  69. </el-collapse>
  70. </div>
  71. </div>
  72. </div>
  73. </template>
  74. <script lang="ts">
  75. import { defineComponent, ref, watch } from 'vue';
  76. import { ElPagination, ElDescriptions, ElCollapse, ElCollapseItem } from 'element-plus';
  77. import { DATA_CRAWLING_BASE_URL } from "@/config";
  78. import axios from 'axios';
  79. import { KnowledgeType, NeedStoreEnum, QueryTask, QueryTypeEnum, QuestionTask, TaskStatusEnum } from '@/types/DataCrawing';
  80. export default defineComponent({
  81. name: 'DataCrawling',
  82. components: {
  83. ElPagination,
  84. ElDescriptions,
  85. ElCollapse,
  86. ElCollapseItem,
  87. },
  88. setup() {
  89. const questionTasks = ref<QuestionTask[]>([]);
  90. const pageNumber = ref(1);
  91. const pageSize = ref(10);
  92. const total = ref(0);
  93. const loading = ref(false);
  94. const selectedTaskId = ref<string | null>(null);
  95. const selectedTask = ref<QuestionTask | null>(null);
  96. const selectedQueries = ref<QueryTask[]>([]);
  97. const activeCollapseName = ref<string | number | null>(null);
  98. // 监听selectedTaskId变化
  99. watch(selectedTaskId, (newVal) => {
  100. selectedTask.value = questionTasks.value.find(task => task.task_id === newVal) || null;
  101. selectedQueries.value = selectedTask.value?.queries || [];
  102. activeCollapseName.value = null; // 重置展开状态
  103. });
  104. const renderTaskStatus = (status: TaskStatusEnum) => {
  105. return TaskStatusEnum[status];
  106. };
  107. const fetchData = async () => {
  108. loading.value = true;
  109. try {
  110. const response = await axios.get(`${DATA_CRAWLING_BASE_URL}/tasks`, {
  111. params: {
  112. page_number: pageNumber.value,
  113. page_size: pageSize.value,
  114. },
  115. });
  116. console.log('Response:', response);
  117. if (response && response.data) {
  118. questionTasks.value = response.data.tasks;
  119. total.value = response.data.total;
  120. selectedTaskId.value = response.data.tasks[0]?.task_id || null;
  121. }
  122. } catch (error) {
  123. console.error('Error fetching data:', error);
  124. } finally {
  125. loading.value = false;
  126. }
  127. };
  128. const handleCurrentChange = (newPage: number) => {
  129. pageNumber.value = newPage;
  130. fetchData();
  131. };
  132. const handleTaskClick = (taskId: string) => {
  133. selectedTaskId.value = taskId;
  134. };
  135. const handleCollapseChange = async (activeName: string | number | (string | number)[]) => {
  136. console.log('activeName:', activeName);
  137. // 查找对应的 query
  138. const query = selectedQueries.value.find(q => q.query === activeName);
  139. // 如果这个 query 已经有数据了,就不再请求
  140. if (query && !query.data && !query.loading) {
  141. query.loading = true;
  142. try {
  143. // 第一步:请求 knowledge-query/data 接口
  144. const queryResponse = await axios.get(`${DATA_CRAWLING_BASE_URL}/knowledge-query/data`, {
  145. params: {
  146. suggest_task_id: query.task_id || selectedTaskId.value,
  147. query: query.query,
  148. },
  149. });
  150. if (queryResponse.data && queryResponse.data.request_id) {
  151. query.request_id = queryResponse.data.request_id;
  152. // 第二步:使用 request_id 请求 knowledge-store/data 接口
  153. const storeResponse = await axios.get(`${DATA_CRAWLING_BASE_URL}/knowledge-store/data`, {
  154. params: {
  155. request_id: query.request_id,
  156. },
  157. });
  158. if (storeResponse.data && storeResponse.data.data) {
  159. query.data = storeResponse.data.data;
  160. }
  161. }
  162. } catch (error) {
  163. console.error('Error fetching query data:', error);
  164. } finally {
  165. query.loading = false;
  166. }
  167. }
  168. };
  169. fetchData();
  170. return {
  171. questionTasks,
  172. pageNumber,
  173. pageSize,
  174. total,
  175. handleCurrentChange,
  176. loading,
  177. selectedTaskId,
  178. selectedTask,
  179. handleTaskClick,
  180. renderTaskStatus,
  181. NeedStoreEnum,
  182. KnowledgeType,
  183. QueryTypeEnum,
  184. selectedQueries,
  185. handleCollapseChange,
  186. };
  187. },
  188. });
  189. </script>
  190. <style scoped>
  191. .app-container {
  192. min-height: calc(100vh - 71px);
  193. background: rgba(138,147,228,0.1);
  194. font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
  195. }
  196. .pagination-container {
  197. box-sizing: border-box;
  198. height: calc(100vh - 90px);
  199. max-height: calc(100vh - 90px);
  200. display: flex;
  201. justify-content: center;
  202. align-items: flex-start;
  203. gap: 10px;
  204. }
  205. .tasks-container {
  206. display: flex;
  207. box-sizing: border-box;
  208. padding-top: 20px;
  209. flex-direction: column;
  210. justify-content: space-between;
  211. align-items: flex-start;
  212. width: 30%;
  213. height: 100%;
  214. overflow-y: auto;
  215. }
  216. .task-detail-container {
  217. box-sizing: border-box;
  218. flex: 1;
  219. padding: 20px;
  220. margin-top: 20px;
  221. margin-right: 20px;
  222. height: calc(100vh - 110px);
  223. max-height: calc(100vh - 110px);
  224. overflow-y: auto;
  225. background: white;
  226. border-radius: 8px;
  227. box-shadow: 0 4px 12px rgba(138, 147, 228, 0.15);
  228. border: 1px solid #e2e8f0;
  229. }
  230. .pagination {
  231. width: 80%;
  232. padding: 0 10px 0 20px;
  233. }
  234. .tasks {
  235. box-sizing: border-box;
  236. width: 100%;
  237. padding: 0 5px 0 20px;
  238. }
  239. .task-item {
  240. box-sizing: border-box;
  241. padding: 16px 20px;
  242. margin-bottom: 12px;
  243. background: white;
  244. border-radius: 8px;
  245. cursor: pointer;
  246. transition: all 0.3s ease;
  247. border: 2px solid transparent;
  248. box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05);
  249. }
  250. .task-item:hover {
  251. border-color: rgba(138, 147, 228, 0.4);
  252. box-shadow: 0 4px 12px rgba(138, 147, 228, 0.15);
  253. transform: translateY(-2px);
  254. }
  255. .task-item-selected {
  256. background: rgba(138, 147, 228, 0.3);
  257. border-color: rgb(138, 147, 228);
  258. box-shadow: 0 4px 12px rgba(138, 147, 228, 0.25);
  259. }
  260. .task-item-selected:hover {
  261. background: rgba(138, 147, 228, 0.15);
  262. }
  263. .task-question {
  264. font-size: 16px;
  265. color: #333;
  266. line-height: 1.6;
  267. font-weight: 500;
  268. }
  269. .query-data-container {
  270. padding: 16px;
  271. min-height: 100px;
  272. }
  273. .query-data-content {
  274. font-size: 14px;
  275. color: #333;
  276. line-height: 1.8;
  277. white-space: pre-wrap;
  278. word-wrap: break-word;
  279. background: #f8f9fa;
  280. padding: 16px;
  281. border-radius: 6px;
  282. border-left: 3px solid rgba(138, 147, 228, 0.6);
  283. }
  284. .query-data-empty {
  285. text-align: center;
  286. color: #999;
  287. font-size: 14px;
  288. padding: 40px;
  289. }
  290. .task-detail-container :deep(.el-descriptions__title) {
  291. font-size: 18px;
  292. line-height: 1.6;
  293. font-weight: 500;
  294. color: #666;
  295. margin: 10px 0 0;
  296. }
  297. .querys-header {
  298. font-size: 18px;
  299. line-height: 1.6;
  300. font-weight: 500;
  301. color: #666;
  302. margin: 25px 0 15px 0;
  303. }
  304. .query-title :deep(.el-collapse-item__header) {
  305. font-size: 15px;
  306. color: #333;
  307. }
  308. </style>