cx-chat 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (404) hide show
  1. package/.cursor/rules/i18n-cn-gloss-comments.mdc +31 -0
  2. package/.cursor/rules/list-page-view-pageconfig.mdc +32 -0
  3. package/.cursor/rules/no-over-defensive-programming.mdc +90 -0
  4. package/.cursor/rules/requirement-description-for-agent.mdc +33 -0
  5. package/.cursor/rules/use-showToast-not-antd-message.mdc +28 -0
  6. package/.docker/Dockerfile +7 -0
  7. package/.env +9 -0
  8. package/.env.development +7 -0
  9. package/.env.production +7 -0
  10. package/.gitlab-ci/docker-build.yaml +28 -0
  11. package/.gitlab-ci/k8s-deploy-dev-master.yaml +42 -0
  12. package/.gitlab-ci/npm-build.yaml +17 -0
  13. package/.gitlab-ci.yml +8 -0
  14. package/.k8s/0-namespace.yaml +6 -0
  15. package/.k8s/1-configmap-web.yaml +7 -0
  16. package/.k8s/1-nginx-conf-dev.yaml +110 -0
  17. package/.k8s/2-deployment.yaml +27 -0
  18. package/.k8s/3-service.yaml +16 -0
  19. package/.k8s/4-ingress-dev.yaml +30 -0
  20. package/.lingma/rules/use-showToast-not-antd-message.md +34 -0
  21. package/.nginx/nginx.conf +52 -0
  22. package/.prettierrc +9 -0
  23. package/README.md +1 -0
  24. package/eslint.config.js +32 -0
  25. package/index.html +13 -0
  26. package/package.json +67 -0
  27. package/postcss.config.js +6 -0
  28. package/public/favicon.ico +0 -0
  29. package/public/vite.svg +1 -0
  30. package/src/App.tsx +96 -0
  31. package/src/_doc/0.docs-overview.md +28 -0
  32. package/src/_doc/cx-ui/0.docs-overview.md +30 -0
  33. package/src/_doc/cx-ui/comp.1.cx-ui-overview.md +82 -0
  34. package/src/_doc/cx-ui/comp.2.cx-modal.md +82 -0
  35. package/src/_doc/cx-ui/comp.3.cx-button.md +89 -0
  36. package/src/_doc/cx-ui/comp.4.cx-form.md +72 -0
  37. package/src/_doc/cx-ui/comp.5.cx-fields.md +76 -0
  38. package/src/_doc/cx-ui/comp.6.cx-tag.md +57 -0
  39. package/src/_doc/cx-ui/comp.7.cx-empty-state.md +29 -0
  40. package/src/_doc/meta/0.docs-overview.md +24 -0
  41. package/src/_doc/meta/comp.1.enum-runtime.md +33 -0
  42. package/src/_doc/meta/comp.2.dict-runtime.md +39 -0
  43. package/src/_doc/router/0.docs-overview.md +14 -0
  44. package/src/_doc/router/guide.1.menu-component-config.md +181 -0
  45. package/src/_doc/router/guide.2.router-auto-registration.md +114 -0
  46. package/src/_doc/table-view/0.docs-overview.md +30 -0
  47. package/src/_doc/table-view/comp.1.table-view.md +542 -0
  48. package/src/_doc/table-view/props.1.create-table-view-config.md +193 -0
  49. package/src/_doc/table-view/props.2.table-view-search-fields.md +106 -0
  50. package/src/api/_mock/README.md +340 -0
  51. package/src/api/_mock/api.ts +1642 -0
  52. package/src/api/_mock/bundle-shim.ts +16 -0
  53. package/src/api/_mock/handler-shim.ts +6 -0
  54. package/src/api/_mock/handler.ts +458 -0
  55. package/src/api/_mock/index.ts +711 -0
  56. package/src/api/_mock/interceptor.ts +15 -0
  57. package/src/api/_mock/mod.ts +12 -0
  58. package/src/api/_mock/utils.ts +65 -0
  59. package/src/api/base/memory.js +24 -0
  60. package/src/api/chat.js +210 -0
  61. package/src/api/common/auth.js +70 -0
  62. package/src/api/menus/business-rules.js +76 -0
  63. package/src/api/menus/feedback.js +102 -0
  64. package/src/api/menus/knowledge.js +159 -0
  65. package/src/api/menus/model-metadata/manage.js +70 -0
  66. package/src/api/menus/model-metadata/role.js +50 -0
  67. package/src/api/menus/model-metadata/training-detail-mock-data.js +569 -0
  68. package/src/api/menus/model-metadata/training.js +28 -0
  69. package/src/api/menus/skill.js +40 -0
  70. package/src/api/system/agent-config.js +16 -0
  71. package/src/api/system/department.js +94 -0
  72. package/src/api/system/dict.js +86 -0
  73. package/src/api/system/menu.js +37 -0
  74. package/src/api/system/permission.js +26 -0
  75. package/src/api/system/role.js +34 -0
  76. package/src/api/system/sys-config.js +16 -0
  77. package/src/api/system/sys-log.js +17 -0
  78. package/src/api/system/user.js +75 -0
  79. package/src/api/upload.js +39 -0
  80. package/src/assets/react.svg +1 -0
  81. package/src/components/auth/current-user-avatar.tsx +77 -0
  82. package/src/components/common/code-view.tsx +149 -0
  83. package/src/components/common/detail-link.tsx +67 -0
  84. package/src/components/common/error-boundary.tsx +98 -0
  85. package/src/components/common/language-switcher.tsx +91 -0
  86. package/src/components/common/lite-table/index.tsx +135 -0
  87. package/src/components/common/md-editor.tsx +126 -0
  88. package/src/components/common/modal/confirm-dialog.tsx +113 -0
  89. package/src/components/common/modal/dep-user-select-multi.tsx +324 -0
  90. package/src/components/common/modal/dep-user-select.tsx +249 -0
  91. package/src/components/common/modal/user-select-multi.tsx +266 -0
  92. package/src/components/common/pagination.tsx +472 -0
  93. package/src/components/common/path.tsx +175 -0
  94. package/src/components/common/system-logo-mark.tsx +48 -0
  95. package/src/components/cx-ui/button/index.less +208 -0
  96. package/src/components/cx-ui/button/index.tsx +611 -0
  97. package/src/components/cx-ui/checkbox/index.tsx +78 -0
  98. package/src/components/cx-ui/date-picker/index.less +17 -0
  99. package/src/components/cx-ui/date-picker/index.tsx +193 -0
  100. package/src/components/cx-ui/drawer/index.tsx +47 -0
  101. package/src/components/cx-ui/empty-state/index.tsx +20 -0
  102. package/src/components/cx-ui/floating-shell/CxFloatingShell.tsx +89 -0
  103. package/src/components/cx-ui/floating-shell/cx-floating-shell.less +283 -0
  104. package/src/components/cx-ui/floating-shell/has-floating-value.ts +41 -0
  105. package/src/components/cx-ui/form/CxForm.tsx +15 -0
  106. package/src/components/cx-ui/form/index.tsx +20 -0
  107. package/src/components/cx-ui/form-item/index.less +26 -0
  108. package/src/components/cx-ui/form-item/index.tsx +36 -0
  109. package/src/components/cx-ui/index.ts +70 -0
  110. package/src/components/cx-ui/input/auto-complete.tsx +134 -0
  111. package/src/components/cx-ui/input/index.tsx +259 -0
  112. package/src/components/cx-ui/input-number/index.jsx +66 -0
  113. package/src/components/cx-ui/modal/index.jsx +212 -0
  114. package/src/components/cx-ui/modal/index.less +144 -0
  115. package/src/components/cx-ui/modal/useCxModal.ts +125 -0
  116. package/src/components/cx-ui/multi-select/index.jsx +74 -0
  117. package/src/components/cx-ui/multi-select/index.less +40 -0
  118. package/src/components/cx-ui/multi-select/index2.tsx +361 -0
  119. package/src/components/cx-ui/radio/index.tsx +33 -0
  120. package/src/components/cx-ui/range-picker/index.less +65 -0
  121. package/src/components/cx-ui/range-picker/index.tsx +219 -0
  122. package/src/components/cx-ui/select/index.less +34 -0
  123. package/src/components/cx-ui/select/index.tsx +196 -0
  124. package/src/components/cx-ui/skeleton/index.tsx +12 -0
  125. package/src/components/cx-ui/steps/index.tsx +14 -0
  126. package/src/components/cx-ui/styles/_tokens.less +79 -0
  127. package/src/components/cx-ui/styles/index.less +246 -0
  128. package/src/components/cx-ui/switch/index.less +106 -0
  129. package/src/components/cx-ui/switch/index.tsx +120 -0
  130. package/src/components/cx-ui/table/index.less +160 -0
  131. package/src/components/cx-ui/table/index.tsx +152 -0
  132. package/src/components/cx-ui/tabs/index.less +15 -0
  133. package/src/components/cx-ui/tabs/index.tsx +34 -0
  134. package/src/components/cx-ui/tag/index.less +51 -0
  135. package/src/components/cx-ui/tag/index.tsx +140 -0
  136. package/src/components/cx-ui/timeline/index.tsx +14 -0
  137. package/src/components/cx-ui/tooltip/index.tsx +67 -0
  138. package/src/components/cx-ui/tree/index.tsx +193 -0
  139. package/src/components/cx-ui/tree-select/index.jsx +91 -0
  140. package/src/components/cx-ui/tree-select/index.less +27 -0
  141. package/src/components/cx-ui/upload-file/index.less +223 -0
  142. package/src/components/cx-ui/upload-file/index.tsx +640 -0
  143. package/src/components/cx-ui/upload-img/index.tsx +291 -0
  144. package/src/components/layout/components/Header.tsx +216 -0
  145. package/src/components/layout/components/Sidebar.tsx +717 -0
  146. package/src/components/layout/index.tsx +95 -0
  147. package/src/components/table-view/components/search-area.tsx +411 -0
  148. package/src/components/table-view/components/table-view-config.tsx +528 -0
  149. package/src/components/table-view/components/table-view.types.ts +478 -0
  150. package/src/components/table-view/components/tree-api-normalize.ts +38 -0
  151. package/src/components/table-view/components/tree-data-annotate.ts +31 -0
  152. package/src/components/table-view/components/tree-sidebar.tsx +74 -0
  153. package/src/components/table-view/index.tsx +61 -0
  154. package/src/components/table-view/list-page-view.tsx +1049 -0
  155. package/src/components/table-view/select-table-view.tsx +1094 -0
  156. package/src/components/table-view/styles/select-table-view.less +51 -0
  157. package/src/config/default-system-name.ts +9 -0
  158. package/src/config/system.ts +165 -0
  159. package/src/constants/countryCodes.ts +3 -0
  160. package/src/contexts/AuthContext.tsx +256 -0
  161. package/src/contexts/ChatContext.tsx +839 -0
  162. package/src/contexts/MenuContext.tsx +62 -0
  163. package/src/contexts/ToastContext.tsx +181 -0
  164. package/src/hooks/useCopyToClipboard.ts +47 -0
  165. package/src/hooks/useModalSubmit.ts +104 -0
  166. package/src/hooks/useRouter.ts +240 -0
  167. package/src/hooks/useStepForm.ts +46 -0
  168. package/src/hooks/useStickyHeader.ts +42 -0
  169. package/src/hooks/useThreadActions.ts +105 -0
  170. package/src/hooks/useUserPreferences.ts +101 -0
  171. package/src/http/axios.js +372 -0
  172. package/src/http/mock.interceptor.ts +9 -0
  173. package/src/http/obfuscationKey.ts +41 -0
  174. package/src/i18n.ts +60 -0
  175. package/src/index.js +1 -0
  176. package/src/index.less +169 -0
  177. package/src/locales/en/auth.ts +70 -0
  178. package/src/locales/en/base/memory.ts +28 -0
  179. package/src/locales/en/base/settings.ts +41 -0
  180. package/src/locales/en/chat.ts +40 -0
  181. package/src/locales/en/common.ts +173 -0
  182. package/src/locales/en/enum.ts +27 -0
  183. package/src/locales/en/menus/business-rules.ts +48 -0
  184. package/src/locales/en/menus/feedback.ts +62 -0
  185. package/src/locales/en/menus/knowledge.ts +120 -0
  186. package/src/locales/en/menus/model-metadata/index.ts +10 -0
  187. package/src/locales/en/menus/model-metadata/manage.ts +151 -0
  188. package/src/locales/en/menus/model-metadata/role.ts +48 -0
  189. package/src/locales/en/menus/model-metadata/training.ts +65 -0
  190. package/src/locales/en/menus/skill.ts +34 -0
  191. package/src/locales/en/system/agent-config.ts +34 -0
  192. package/src/locales/en/system/department.ts +68 -0
  193. package/src/locales/en/system/dict.ts +44 -0
  194. package/src/locales/en/system/menu.ts +45 -0
  195. package/src/locales/en/system/permission.ts +89 -0
  196. package/src/locales/en/system/role.ts +25 -0
  197. package/src/locales/en/system/sys-config.ts +33 -0
  198. package/src/locales/en/system/sys-log.ts +38 -0
  199. package/src/locales/en/system/user.ts +113 -0
  200. package/src/locales/en.ts +68 -0
  201. package/src/locales/zh/auth.ts +70 -0
  202. package/src/locales/zh/base/memory.ts +29 -0
  203. package/src/locales/zh/base/settings.ts +41 -0
  204. package/src/locales/zh/chat.ts +47 -0
  205. package/src/locales/zh/common.ts +178 -0
  206. package/src/locales/zh/enum.ts +28 -0
  207. package/src/locales/zh/menus/business-rules.ts +47 -0
  208. package/src/locales/zh/menus/feedback.ts +62 -0
  209. package/src/locales/zh/menus/knowledge.ts +117 -0
  210. package/src/locales/zh/menus/model-metadata/index.ts +10 -0
  211. package/src/locales/zh/menus/model-metadata/manage.ts +151 -0
  212. package/src/locales/zh/menus/model-metadata/role.ts +47 -0
  213. package/src/locales/zh/menus/model-metadata/training.ts +64 -0
  214. package/src/locales/zh/menus/skill.ts +34 -0
  215. package/src/locales/zh/system/agent-config.ts +33 -0
  216. package/src/locales/zh/system/department.ts +69 -0
  217. package/src/locales/zh/system/dict.ts +44 -0
  218. package/src/locales/zh/system/menu.ts +47 -0
  219. package/src/locales/zh/system/permission.ts +94 -0
  220. package/src/locales/zh/system/role.ts +25 -0
  221. package/src/locales/zh/system/sys-config.ts +32 -0
  222. package/src/locales/zh/system/sys-log.ts +38 -0
  223. package/src/locales/zh/system/user.ts +114 -0
  224. package/src/locales/zh.ts +67 -0
  225. package/src/main.tsx +50 -0
  226. package/src/meta/const/index.ts +40 -0
  227. package/src/meta/index-dict.ts +56 -0
  228. package/src/meta/index-enum.ts +95 -0
  229. package/src/meta/index.ts +14 -0
  230. package/src/meta/module/dict-data/runtime.ts +199 -0
  231. package/src/meta/module/dict-data/types.ts +17 -0
  232. package/src/meta/module/enum-data/runtime.ts +75 -0
  233. package/src/meta/module/enum-data/types.ts +18 -0
  234. package/src/router/index.tsx +312 -0
  235. package/src/styles/AntdThemeProvider.tsx +40 -0
  236. package/src/styles/antd-theme.ts +20 -0
  237. package/src/styles/global.less +107 -0
  238. package/src/styles/variable.less +103 -0
  239. package/src/types/feedback.ts +43 -0
  240. package/src/types/index.ts +85 -0
  241. package/src/types/menu.ts +43 -0
  242. package/src/utils/aesUtil.ts +123 -0
  243. package/src/utils/chatUtils.ts +524 -0
  244. package/src/utils/cn.ts +6 -0
  245. package/src/utils/crypto.ts +164 -0
  246. package/src/utils/date.ts +72 -0
  247. package/src/utils/file-icons.tsx +79 -0
  248. package/src/utils/index.ts +168 -0
  249. package/src/utils/markdown-math-plugins.ts +21 -0
  250. package/src/utils/menuI18n.ts +305 -0
  251. package/src/utils/menuRouteRegistry.ts +78 -0
  252. package/src/utils/permission-crud.ts +147 -0
  253. package/src/utils/routeConfig.ts +350 -0
  254. package/src/utils/storage.ts +135 -0
  255. package/src/utils/toastBridge.ts +26 -0
  256. package/src/utils/url.ts +38 -0
  257. package/src/utils/validation.ts +16 -0
  258. package/src/views/auth/auth-code/index.less +169 -0
  259. package/src/views/auth/auth-code/index.module.less +174 -0
  260. package/src/views/auth/auth-code/index.tsx +233 -0
  261. package/src/views/auth/login.tsx +498 -0
  262. package/src/views/auth/register.tsx +388 -0
  263. package/src/views/base/memory/index.tsx +136 -0
  264. package/src/views/base/memory/modal/detail-modal.tsx +89 -0
  265. package/src/views/base/memory/modal/submit-modal.tsx +134 -0
  266. package/src/views/base/settings/index.tsx +657 -0
  267. package/src/views/chat/chat.less +323 -0
  268. package/src/views/chat/components/chat-input.tsx +298 -0
  269. package/src/views/chat/components/header-thread-title.tsx +210 -0
  270. package/src/views/chat/components/message-list/content-answer.tsx +100 -0
  271. package/src/views/chat/components/message-list/content-question.tsx +18 -0
  272. package/src/views/chat/components/message-list/index.tsx +520 -0
  273. package/src/views/chat/components/message-list/message-item.tsx +199 -0
  274. package/src/views/chat/components/message-list/preparation-demo-items.ts +147 -0
  275. package/src/views/chat/components/message-list/preparation-steps.tsx +506 -0
  276. package/src/views/chat/components/message-list/suggestion-list.tsx +36 -0
  277. package/src/views/chat/components/message-list/thinking-process.tsx +49 -0
  278. package/src/views/chat/components/message-list/toolbar.tsx +224 -0
  279. package/src/views/chat/components/message-list/use-message-list-scroll.ts +214 -0
  280. package/src/views/chat/components/references-knowledge/context.tsx +57 -0
  281. package/src/views/chat/components/references-knowledge/index.ts +9 -0
  282. package/src/views/chat/components/references-knowledge/modal/knowledge-detail-drawer.tsx +556 -0
  283. package/src/views/chat/components/references-knowledge/modal/knowledge-doc-detail-drawer.tsx +529 -0
  284. package/src/views/chat/components/references-knowledge/panel.tsx +115 -0
  285. package/src/views/chat/hooks/use-chat-common.ts +19 -0
  286. package/src/views/chat/index-session.tsx +647 -0
  287. package/src/views/chat/index.tsx +127 -0
  288. package/src/views/page-error/401.tsx +56 -0
  289. package/src/views/page-error/404.tsx +56 -0
  290. package/src/views/page-menus/business-rules/index.tsx +376 -0
  291. package/src/views/page-menus/business-rules/modal/detail-modal.tsx +186 -0
  292. package/src/views/page-menus/business-rules/modal/scope-modal.tsx +272 -0
  293. package/src/views/page-menus/business-rules/modal/submit-modal.tsx +142 -0
  294. package/src/views/page-menus/feedback/components/feedback-dataset-list.tsx +471 -0
  295. package/src/views/page-menus/feedback/index.tsx +166 -0
  296. package/src/views/page-menus/feedback/modal/export-feedback-modal.tsx +367 -0
  297. package/src/views/page-menus/knowledge/components/doc-editor-by-type.tsx +32 -0
  298. package/src/views/page-menus/knowledge/components/doc-editor-type-file.tsx +330 -0
  299. package/src/views/page-menus/knowledge/detail.tsx +600 -0
  300. package/src/views/page-menus/knowledge/index.tsx +337 -0
  301. package/src/views/page-menus/knowledge/modal/detail-modal.tsx +618 -0
  302. package/src/views/page-menus/knowledge/modal/doc-detail-modal.tsx +550 -0
  303. package/src/views/page-menus/knowledge/modal/doc-parse.ts +99 -0
  304. package/src/views/page-menus/knowledge/modal/doc-submit-modal.tsx +349 -0
  305. package/src/views/page-menus/knowledge/modal/doc-type-picker-modal.tsx +88 -0
  306. package/src/views/page-menus/knowledge/modal/knowledge-user-select-modal.tsx +283 -0
  307. package/src/views/page-menus/knowledge/modal/submit-modal.tsx +179 -0
  308. package/src/views/page-menus/model-metadata/manage/components/metadata-detail-schema-tab.tsx +114 -0
  309. package/src/views/page-menus/model-metadata/manage/components/step1-basic-info.tsx +232 -0
  310. package/src/views/page-menus/model-metadata/manage/components/step2-schema.tsx +316 -0
  311. package/src/views/page-menus/model-metadata/manage/components/step3-permissions.tsx +134 -0
  312. package/src/views/page-menus/model-metadata/manage/components/step4-documents.tsx +134 -0
  313. package/src/views/page-menus/model-metadata/manage/components/step5-example-sql.tsx +101 -0
  314. package/src/views/page-menus/model-metadata/manage/components/submit-add.tsx +338 -0
  315. package/src/views/page-menus/model-metadata/manage/components/submit-edit.tsx +276 -0
  316. package/src/views/page-menus/model-metadata/manage/detail.tsx +298 -0
  317. package/src/views/page-menus/model-metadata/manage/hooks/model-metadata-submit-shared.ts +113 -0
  318. package/src/views/page-menus/model-metadata/manage/hooks/use-model-metadata-item-state.ts +20 -0
  319. package/src/views/page-menus/model-metadata/manage/index.tsx +304 -0
  320. package/src/views/page-menus/model-metadata/manage/modal/components/table-schema.ts +374 -0
  321. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-detail-tabs.tsx +151 -0
  322. package/src/views/page-menus/model-metadata/manage/modal/components/use-table-submit-tabs.tsx +423 -0
  323. package/src/views/page-menus/model-metadata/manage/modal/detail-modal.tsx +218 -0
  324. package/src/views/page-menus/model-metadata/manage/modal/submit-modal.tsx +261 -0
  325. package/src/views/page-menus/model-metadata/manage/modal/table-detail-modal.tsx +196 -0
  326. package/src/views/page-menus/model-metadata/manage/modal/table-submit-modal.tsx +229 -0
  327. package/src/views/page-menus/model-metadata/manage/submit.tsx +31 -0
  328. package/src/views/page-menus/model-metadata/role/index.tsx +207 -0
  329. package/src/views/page-menus/model-metadata/role/modal/detail-modal.tsx +97 -0
  330. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-modal.tsx +254 -0
  331. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-panel.tsx +393 -0
  332. package/src/views/page-menus/model-metadata/role/modal/role-assign-users-utils.ts +120 -0
  333. package/src/views/page-menus/model-metadata/role/modal/role-permission-assign-panel.tsx +698 -0
  334. package/src/views/page-menus/model-metadata/role/modal/role-permission-modal.tsx +237 -0
  335. package/src/views/page-menus/model-metadata/role/modal/submit-modal.tsx +135 -0
  336. package/src/views/page-menus/model-metadata/training/components/detail-records/index.ts +4 -0
  337. package/src/views/page-menus/model-metadata/training/components/detail-records/node-card.tsx +72 -0
  338. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-lines.ts +196 -0
  339. package/src/views/page-menus/model-metadata/training/components/detail-records/summary-list.tsx +153 -0
  340. package/src/views/page-menus/model-metadata/training/components/detail-records/timeline.tsx +103 -0
  341. package/src/views/page-menus/model-metadata/training/components/detail-records/types.ts +82 -0
  342. package/src/views/page-menus/model-metadata/training/detail.tsx +159 -0
  343. package/src/views/page-menus/model-metadata/training/index.tsx +236 -0
  344. package/src/views/page-menus/model-metadata/training/modal/update-detail-modal.tsx +154 -0
  345. package/src/views/page-menus/skill/index.tsx +201 -0
  346. package/src/views/page-menus/skill/modal/detail-modal.tsx +156 -0
  347. package/src/views/page-menus/skill/modal/submit-modal.tsx +214 -0
  348. package/src/views/page-system/agent-config/index.tsx +370 -0
  349. package/src/views/page-system/department/departmentFormShared.ts +36 -0
  350. package/src/views/page-system/department/index.tsx +541 -0
  351. package/src/views/page-system/department/modal/detail-modal.tsx +94 -0
  352. package/src/views/page-system/department/modal/member-role-modal.tsx +128 -0
  353. package/src/views/page-system/department/modal/submit-modal.tsx +265 -0
  354. package/src/views/page-system/dict/index.tsx +440 -0
  355. package/src/views/page-system/dict/modal/cate-submit-modal.tsx +315 -0
  356. package/src/views/page-system/dict/modal/submit-modal.tsx +184 -0
  357. package/src/views/page-system/logs/components/index.ts +3 -0
  358. package/src/views/page-system/logs/components/log-message-demo.tsx +30 -0
  359. package/src/views/page-system/logs/components/log-message-stream.ts +184 -0
  360. package/src/views/page-system/logs/components/message-list/content-answer.tsx +100 -0
  361. package/src/views/page-system/logs/components/message-list/content-question.tsx +18 -0
  362. package/src/views/page-system/logs/components/message-list/index.tsx +515 -0
  363. package/src/views/page-system/logs/components/message-list/message-item.tsx +193 -0
  364. package/src/views/page-system/logs/components/message-list/preparation-demo-items.ts +147 -0
  365. package/src/views/page-system/logs/components/message-list/preparation-steps.tsx +506 -0
  366. package/src/views/page-system/logs/components/message-list/suggestion-list.tsx +36 -0
  367. package/src/views/page-system/logs/components/message-list/thinking-process.tsx +49 -0
  368. package/src/views/page-system/logs/components/message-list/toolbar.tsx +134 -0
  369. package/src/views/page-system/logs/components/message-list/use-message-list-scroll.ts +214 -0
  370. package/src/views/page-system/logs/components/message-modal.tsx +239 -0
  371. package/src/views/page-system/logs/index.tsx +132 -0
  372. package/src/views/page-system/logs/modal/detail-modal.tsx +157 -0
  373. package/src/views/page-system/menu/components/menuFormShared.ts +283 -0
  374. package/src/views/page-system/menu/index.less +12 -0
  375. package/src/views/page-system/menu/index.tsx +410 -0
  376. package/src/views/page-system/menu/modal/icon-modal.less +51 -0
  377. package/src/views/page-system/menu/modal/icon-modal.tsx +87 -0
  378. package/src/views/page-system/menu/modal/submit-modal.tsx +263 -0
  379. package/src/views/page-system/permission/index.tsx +562 -0
  380. package/src/views/page-system/permission/modal/detail-modal.tsx +179 -0
  381. package/src/views/page-system/permission/modal/submit-modal.less +146 -0
  382. package/src/views/page-system/permission/modal/submit-modal.tsx +650 -0
  383. package/src/views/page-system/role/index.tsx +163 -0
  384. package/src/views/page-system/role/modal/detail-modal.tsx +127 -0
  385. package/src/views/page-system/role/modal/permission-assign-group-rules.ts +86 -0
  386. package/src/views/page-system/role/modal/permission-modal.tsx +111 -0
  387. package/src/views/page-system/role/modal/role-modal-shell-styles.ts +21 -0
  388. package/src/views/page-system/role/modal/role-permission-assign-panel.tsx +916 -0
  389. package/src/views/page-system/role/modal/role-permission-assign-shared.ts +1047 -0
  390. package/src/views/page-system/role/modal/submit-modal.tsx +193 -0
  391. package/src/views/page-system/sys-config/index.tsx +294 -0
  392. package/src/views/page-system/user/components/user-role-column.tsx +87 -0
  393. package/src/views/page-system/user/index.tsx +439 -0
  394. package/src/views/page-system/user/modal/assign-roles-modal.tsx +389 -0
  395. package/src/views/page-system/user/modal/detail-modal.tsx +72 -0
  396. package/src/views/page-system/user/modal/modal-style/submit-modal.less +40 -0
  397. package/src/views/page-system/user/modal/submit-modal.less +40 -0
  398. package/src/views/page-system/user/modal/submit-modal.tsx +287 -0
  399. package/src/views/page-system/user/userFormShared.ts +51 -0
  400. package/tailwind.config.js +17 -0
  401. package/tsconfig.app.json +48 -0
  402. package/tsconfig.json +11 -0
  403. package/tsconfig.node.json +26 -0
  404. package/vite.config.ts +264 -0
@@ -0,0 +1,839 @@
1
+ import {
2
+ CHAT_STREAM_HANDLED_JSON_ERROR,
3
+ getChatTitleAPI,
4
+ postChatStreamRequest,
5
+ updateChatTitleAPI,
6
+ } from '@/api/chat';
7
+ import {
8
+ markMessagePreparationDone,
9
+ mergeMessagePreparationDelta,
10
+ omitMessagePreparationForPersist,
11
+ resolveMessageTurnIdForFeedback,
12
+ resolveThreadDisplayTitle,
13
+ } from '@/utils/chatUtils';
14
+ import { useQueryClient, type QueryClient } from '@tanstack/react-query';
15
+ import React, { createContext, useCallback, useContext, useRef, useState } from 'react';
16
+ import { flushSync } from 'react-dom';
17
+ import { useTranslation } from 'react-i18next';
18
+ import type { Message, Thread } from '../types';
19
+ import { useAuth } from './AuthContext';
20
+ import { useToast } from './ToastContext';
21
+
22
+ export interface ChatStreamSendOptions {
23
+ enableNetwork?: boolean;
24
+ enableKnowledgeBase?: boolean;
25
+ enableDataAnalysis?: boolean;
26
+ }
27
+
28
+ /** 会话流式:联网 / 知识库 / 数据分析 开关对应权限码(与后端菜单权限一致) */
29
+ export const CHAT_PERM_ENABLE_NETWORK = 'system:chat:enable-network';
30
+ export const CHAT_PERM_ENABLE_KNOWLEDGE = 'system:chat:enable-knowledge';
31
+ export const CHAT_PERM_ENABLE_DATA_ANALYSIS = 'system:chat:enable-data-analysis';
32
+
33
+ /** 无权限则强制为 false,避免仅靠前端开关或 localStorage 绕过 */
34
+ export function clampChatStreamFeatureFlags(
35
+ options: ChatStreamSendOptions | undefined,
36
+ hasPermission: (code: string) => boolean,
37
+ ): {
38
+ enableNetwork: boolean;
39
+ enableKnowledgeBase: boolean;
40
+ enableDataAnalysis: boolean;
41
+ } {
42
+ return {
43
+ enableNetwork:
44
+ Boolean(options?.enableNetwork) &&
45
+ hasPermission(CHAT_PERM_ENABLE_NETWORK),
46
+ enableKnowledgeBase:
47
+ Boolean(options?.enableKnowledgeBase) &&
48
+ hasPermission(CHAT_PERM_ENABLE_KNOWLEDGE),
49
+ enableDataAnalysis:
50
+ Boolean(options?.enableDataAnalysis) &&
51
+ hasPermission(CHAT_PERM_ENABLE_DATA_ANALYSIS),
52
+ };
53
+ }
54
+
55
+ interface ChatContextType {
56
+ activeThreadId: string | null;
57
+ isStreaming: boolean;
58
+ streamingContent: Message | null;
59
+ /** 新会话乐观插入的列表行;流式结束并刷新列表后清除,供侧栏在接口未返回前合并展示 */
60
+ streamingPlaceholderThread: Thread | null;
61
+ sendMessage: (
62
+ content: string,
63
+ threadId: string,
64
+ options?: ChatStreamSendOptions,
65
+ ) => Promise<void>;
66
+ /** 按 turn 重新生成助手回复:POST /chat/stream/regenerate */
67
+ regenerateMessage: (
68
+ sessionId: string,
69
+ turnId: string,
70
+ options?: ChatStreamSendOptions,
71
+ ) => Promise<void>;
72
+ stopStream: () => Promise<void>;
73
+ resetStream: () => void;
74
+ }
75
+
76
+ /** 会话列表接口落库略晚于首条消息:流结束后再延迟拉列表,降低「新会话被刷掉」概率 */
77
+ const THREADS_LIST_REFETCH_DELAY_MS = 600
78
+
79
+ /** 不重拉整表:仅把当前会话置顶并更新 updated_at,避免回答结束后侧栏整表刷新闪动 */
80
+ function bumpThreadToTopInCache(queryClient: QueryClient, threadId: string) {
81
+ queryClient.setQueryData<Thread[]>(['threads'], (old) => {
82
+ const list = old ?? [];
83
+ const idx = list.findIndex((t) => t.id === threadId);
84
+ if (idx < 0) return list;
85
+ const cur = list[idx];
86
+ const next: Thread = {
87
+ ...cur,
88
+ updated_at: new Date().toISOString(),
89
+ };
90
+ return [next, ...list.filter((_, i) => i !== idx)];
91
+ });
92
+ }
93
+
94
+ const ChatContext = createContext<ChatContextType | undefined>(undefined);
95
+
96
+ function pickGeneratedChatTitle(res: unknown): string {
97
+ const r = res as Record<string, unknown> | null | undefined
98
+ const d = r?.data as Record<string, unknown> | undefined
99
+ const nested = d?.data as Record<string, unknown> | undefined
100
+ const title =
101
+ (typeof nested?.title === 'string' && nested.title) ||
102
+ (typeof d?.title === 'string' && d.title) ||
103
+ (typeof r?.title === 'string' && r.title) ||
104
+ ''
105
+ return String(title).trim()
106
+ }
107
+
108
+ type SseStreamRefs = {
109
+ thisStreamGen: number;
110
+ streamGenerationRef: React.MutableRefObject<number>;
111
+ setStreamingContent: React.Dispatch<React.SetStateAction<Message | null>>;
112
+ setIsStreaming: React.Dispatch<React.SetStateAction<boolean>>;
113
+ streamingContentRef: React.MutableRefObject<Message | null>;
114
+ /** 发送新消息时:message_start 拿到后端 turn_id 后,把乐观插入的最后一条用户消息 id 同步为 `{turnId}-10` */
115
+ syncLastUserMessageTurnId?: (turnKey: string) => void;
116
+ };
117
+
118
+ /** 消费流式接口返回的 SSE(data: JSON 行),更新 streamingContent / isStreaming */
119
+ async function readSseStreamBody(
120
+ response: Response,
121
+ refs: SseStreamRefs,
122
+ ): Promise<void> {
123
+ if (!response.body) throw new Error('No response body');
124
+ const reader = response.body.getReader();
125
+ const decoder = new TextDecoder();
126
+ let startBuffer = '';
127
+ const {
128
+ thisStreamGen,
129
+ streamGenerationRef,
130
+ setStreamingContent,
131
+ setIsStreaming,
132
+ streamingContentRef,
133
+ syncLastUserMessageTurnId,
134
+ } = refs;
135
+
136
+ while (true) {
137
+ const { done, value } = await reader.read();
138
+ if (done) break;
139
+
140
+ const chunk = decoder.decode(value, { stream: true });
141
+ startBuffer += chunk;
142
+ const lines = startBuffer.split('\n');
143
+ startBuffer = lines.pop() || '';
144
+
145
+ for (const line of lines) {
146
+ const trimmed = line.replace(/\r$/, '').trim();
147
+ if (!trimmed.startsWith('data:')) continue;
148
+
149
+ const dataStr = trimmed.slice(5).trimStart();
150
+
151
+ try {
152
+ const parsedData: Record<string, unknown> = JSON.parse(dataStr);
153
+ if (thisStreamGen !== streamGenerationRef.current) {
154
+ continue;
155
+ }
156
+ const eventType = parsedData.type as string | undefined;
157
+
158
+ if (eventType === 'message_start') {
159
+ const turnId = parsedData.turn_id;
160
+ const turnKey =
161
+ turnId != null && String(turnId) ? String(turnId) : '';
162
+ setStreamingContent((prev) => {
163
+ if (!prev) return prev;
164
+ const nextId = turnKey ? `${turnKey}-20` : prev.id;
165
+ const updated = {
166
+ ...prev,
167
+ id: nextId,
168
+ turn_id: turnKey || prev.turn_id,
169
+ };
170
+ streamingContentRef.current = updated;
171
+ return updated;
172
+ });
173
+ if (turnKey) {
174
+ syncLastUserMessageTurnId?.(turnKey);
175
+ }
176
+ continue;
177
+ }
178
+
179
+ if (eventType === 'preparation') {
180
+ const delta = parsedData.delta as
181
+ | Record<string, unknown>
182
+ | undefined;
183
+ setStreamingContent((prev) => {
184
+ if (!prev) return null;
185
+ const preparation_items = mergeMessagePreparationDelta(
186
+ prev.preparation_items,
187
+ delta,
188
+ );
189
+ const updated = { ...prev, preparation_items };
190
+ streamingContentRef.current = updated;
191
+ return updated;
192
+ });
193
+ continue;
194
+ }
195
+
196
+ if (eventType === 'content_block_delta') {
197
+ const delta = parsedData.delta as
198
+ | { text?: string }
199
+ | undefined;
200
+ const piece = delta?.text ?? '';
201
+ if (piece) {
202
+ flushSync(() => {
203
+ setStreamingContent((prev) => {
204
+ if (!prev) return null;
205
+ const updated = {
206
+ ...prev,
207
+ content: prev.content + piece,
208
+ };
209
+ streamingContentRef.current = updated;
210
+ return updated;
211
+ });
212
+ });
213
+ }
214
+ continue;
215
+ }
216
+
217
+ if (eventType === 'content_block_stop') {
218
+ setStreamingContent((prev) => {
219
+ if (!prev) return prev;
220
+ const updated = markMessagePreparationDone({
221
+ ...prev,
222
+ is_streaming: false,
223
+ });
224
+ streamingContentRef.current = updated;
225
+ return updated;
226
+ });
227
+ setIsStreaming(false);
228
+ continue;
229
+ }
230
+
231
+ if (
232
+ eventType === 'suggested_replies' &&
233
+ Array.isArray(parsedData.items)
234
+ ) {
235
+ setStreamingContent((prev) => {
236
+ if (!prev) return null;
237
+ const updated = {
238
+ ...prev,
239
+ options: parsedData.items as string[],
240
+ };
241
+ streamingContentRef.current = updated;
242
+ return updated;
243
+ });
244
+ continue;
245
+ }
246
+
247
+ if (
248
+ eventType === 'references_knowledge' &&
249
+ Array.isArray(parsedData.items)
250
+ ) {
251
+ setStreamingContent((prev) => {
252
+ if (!prev) return null;
253
+ const updated = {
254
+ ...prev,
255
+ references_knowledge_items: parsedData.items as any[],
256
+ };
257
+ streamingContentRef.current = updated;
258
+ return updated;
259
+ });
260
+ continue;
261
+ }
262
+
263
+ if (eventType === 'message_end') {
264
+ setStreamingContent((prev) => {
265
+ if (!prev) return prev;
266
+ const updated = markMessagePreparationDone({
267
+ ...prev,
268
+ is_streaming: false,
269
+ });
270
+ streamingContentRef.current = updated;
271
+ return updated;
272
+ });
273
+ continue;
274
+ }
275
+ } catch (e) {
276
+ console.error('解析 SSE 数据错误', e);
277
+ }
278
+ }
279
+ }
280
+ }
281
+
282
+
283
+ export const ChatProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
284
+ const [activeThreadId, setActiveThreadId] = useState<string | null>(null);
285
+ const [isStreaming, setIsStreaming] = useState(false);
286
+ const [streamingContent, setStreamingContent] = useState<Message | null>(null);
287
+ const [streamingPlaceholderThread, setStreamingPlaceholderThread] =
288
+ useState<Thread | null>(null);
289
+ const abortControllerRef = useRef<AbortController | null>(null);
290
+
291
+ // 在异步/事件循环内部访问状态时,如果闭包过期可能需要 refs
292
+ // 但这里我们主要更新状态。streamingContentRef 对清理很有用。
293
+ const streamingContentRef = useRef<Message | null>(null);
294
+ /** 每次发起流式请求递增;用于丢弃被新请求取代的旧 SSE / finally,避免串状态 */
295
+ const streamGenerationRef = useRef(0);
296
+ /** 当前流式请求对应的会话 id(用于被新请求打断时把上一条助手消息写入缓存) */
297
+ const streamingThreadIdRef = useRef<string | null>(null);
298
+
299
+ const queryClient = useQueryClient();
300
+ const { showToast } = useToast();
301
+ const { t } = useTranslation();
302
+ const { hasPermission } = useAuth();
303
+
304
+ const resetStream = useCallback(() => {
305
+ setActiveThreadId(null);
306
+ setIsStreaming(false);
307
+ setStreamingContent(null);
308
+ setStreamingPlaceholderThread(null);
309
+ abortControllerRef.current = null;
310
+ streamingContentRef.current = null;
311
+ streamingThreadIdRef.current = null;
312
+ }, []);
313
+
314
+ const stopStream = useCallback(async () => {
315
+ if (abortControllerRef.current) {
316
+ abortControllerRef.current.abort();
317
+ abortControllerRef.current = null;
318
+
319
+ // 更新 UI 状态为已停止
320
+ setIsStreaming(false);
321
+
322
+ // 如果需要,在本地状态中将消息标记为已中止
323
+ if (streamingContentRef.current) {
324
+ // 在缓存中将消息最终化为已中止 (?) 或让它作为部分内容存在
325
+ // 目前,我们只停止加载动画。`sendMessage` 中的 `handleError` 或 `finally` 块
326
+ // 通常处理“中止”清理,但由于我们手动中止,可能需要触发它。
327
+ }
328
+ }
329
+ }, [activeThreadId]);
330
+
331
+ const sendMessage = useCallback(async (
332
+ message: string,
333
+ threadId: string,
334
+ options?: ChatStreamSendOptions,
335
+ ) => {
336
+ if (isStreaming) {
337
+ console.warn("正在流式传输中,忽略发送请求");
338
+ return;
339
+ }
340
+
341
+ // 新建会话:navigate 后会立刻拉 messages,若与乐观更新并发,返回体会覆盖缓存导致首条用户消息消失
342
+ await queryClient.cancelQueries({ queryKey: ['messages', threadId] })
343
+
344
+ const staleThreadId = streamingThreadIdRef.current;
345
+ const staleAssistant = streamingContentRef.current;
346
+
347
+ streamGenerationRef.current += 1;
348
+ const thisStreamGen = streamGenerationRef.current;
349
+
350
+ if (abortControllerRef.current) {
351
+ abortControllerRef.current.abort();
352
+ abortControllerRef.current = null;
353
+ if (
354
+ staleAssistant?.role === 'assistant' &&
355
+ staleAssistant.content &&
356
+ staleThreadId
357
+ ) {
358
+ const toMerge = omitMessagePreparationForPersist(
359
+ markMessagePreparationDone({
360
+ ...staleAssistant,
361
+ is_streaming: false,
362
+ }),
363
+ );
364
+ queryClient.setQueryData<Message[]>(
365
+ ['messages', staleThreadId],
366
+ (old) => {
367
+ const prev = old ?? [];
368
+ const exists = prev.some((m) => m.id === toMerge.id);
369
+ if (!exists) return [...prev, toMerge];
370
+ return prev.map((m) =>
371
+ m.id === toMerge.id ? toMerge : m,
372
+ );
373
+ },
374
+ );
375
+ }
376
+ }
377
+
378
+ streamingThreadIdRef.current = threadId;
379
+
380
+ setActiveThreadId(threadId);
381
+ setIsStreaming(true);
382
+ setStreamingContent(null);
383
+ streamingContentRef.current = null;
384
+
385
+ const controller = new AbortController();
386
+ abortControllerRef.current = controller;
387
+
388
+ const prevMessages = queryClient.getQueryData<Message[]>(['messages', threadId]) ?? [];
389
+ const isFirstSessionTurn = prevMessages.length === 0;
390
+ let streamFinishedOk = false;
391
+
392
+ // 乐观更新:立即添加用户消息
393
+ const optimisticUserMessage: Message = {
394
+ id: (typeof crypto.randomUUID === 'function') ? crypto.randomUUID() : `u-${Date.now()}`,
395
+ role: 'user',
396
+ content: message,
397
+ created_at: new Date().toISOString(),
398
+ };
399
+
400
+ // 初始化流式消息
401
+ const initialStreamingMessage: Message = {
402
+ id: (typeof crypto.randomUUID === 'function') ? `temp-${crypto.randomUUID()}` : `temp-ai-stream-${Date.now()}`,
403
+ role: 'assistant',
404
+ content: '',
405
+ thoughts: [],
406
+ options: [],
407
+ preparation_items: [],
408
+ is_streaming: true,
409
+ created_at: new Date().toISOString(),
410
+ };
411
+
412
+ setStreamingContent(initialStreamingMessage);
413
+ streamingContentRef.current = initialStreamingMessage;
414
+
415
+ // 追加用户消息,并清除历史里助手消息的搜索建议(避免跟新一轮对话叠在一起)
416
+ queryClient.setQueryData<Message[]>(['messages', threadId], (old) => {
417
+ const prev = old ?? [];
418
+ const cleared = prev.map((m) =>
419
+ m.role === 'assistant' && m.options && m.options.length > 0
420
+ ? { ...m, options: [] as string[] }
421
+ : m,
422
+ );
423
+ return [...cleared, optimisticUserMessage];
424
+ });
425
+
426
+ // 左侧会话列表:立即插入/更新当前会话(避免新会话要等 refetch 才出现)
427
+ // 新会话首轮在助手回答前不用提问当标题,避免先预览再被 getChatTitle 替换造成闪烁
428
+ const now = new Date().toISOString();
429
+ const trimmed = message.trim();
430
+ const preview =
431
+ trimmed.length > 60 ? `${trimmed.slice(0, 60)}…` : trimmed;
432
+ const untitledLabel = t('chat.untitled_session');
433
+ let threadsPlaceholder: Thread | null = null;
434
+ queryClient.setQueryData<Thread[]>(['threads'], (old) => {
435
+ const list = [...(old ?? [])];
436
+ const idx = list.findIndex((t) => t.id === threadId);
437
+ if (idx >= 0) {
438
+ const cur = list[idx];
439
+ const nextTitle = isFirstSessionTurn
440
+ ? resolveThreadDisplayTitle(cur.title, untitledLabel)
441
+ : resolveThreadDisplayTitle(
442
+ preview || cur.title,
443
+ untitledLabel,
444
+ );
445
+ list[idx] = {
446
+ ...cur,
447
+ title: nextTitle,
448
+ updated_at: now,
449
+ };
450
+ threadsPlaceholder = null;
451
+ return list;
452
+ }
453
+ const inserted: Thread = {
454
+ id: threadId,
455
+ title: isFirstSessionTurn
456
+ ? resolveThreadDisplayTitle('', untitledLabel)
457
+ : resolveThreadDisplayTitle(preview, untitledLabel),
458
+ created_at: now,
459
+ updated_at: now,
460
+ user_id: '',
461
+ top_status: 0,
462
+ };
463
+ list.unshift(inserted);
464
+ threadsPlaceholder = inserted;
465
+ return list;
466
+ });
467
+ setStreamingPlaceholderThread(threadsPlaceholder);
468
+
469
+ try {
470
+ const streamFlags = clampChatStreamFeatureFlags(options, hasPermission);
471
+ const response = await postChatStreamRequest(
472
+ {
473
+ message,
474
+ session_id: threadId,
475
+ enableNetwork: streamFlags.enableNetwork,
476
+ enableKnowledgeBase: streamFlags.enableKnowledgeBase,
477
+ enableDataAnalysis: streamFlags.enableDataAnalysis,
478
+ },
479
+ { signal: controller.signal },
480
+ );
481
+
482
+ await readSseStreamBody(response, {
483
+ thisStreamGen,
484
+ streamGenerationRef,
485
+ setStreamingContent,
486
+ setIsStreaming,
487
+ streamingContentRef,
488
+ syncLastUserMessageTurnId: (turnKey: string) => {
489
+ queryClient.setQueryData<Message[]>(
490
+ ['messages', threadId],
491
+ (old) => {
492
+ const prev = old ?? [];
493
+ for (let i = prev.length - 1; i >= 0; i--) {
494
+ if (prev[i].role === 'user') {
495
+ const u = prev[i];
496
+ const next = [...prev];
497
+ next[i] = {
498
+ ...u,
499
+ id: `${turnKey}-10`,
500
+ turn_id: turnKey,
501
+ };
502
+ return next;
503
+ }
504
+ }
505
+ return prev;
506
+ },
507
+ );
508
+ },
509
+ });
510
+
511
+ streamFinishedOk = true;
512
+
513
+ } catch (err: any) {
514
+ if (thisStreamGen !== streamGenerationRef.current) {
515
+ // 已被新请求取代
516
+ } else if (err.name === 'AbortError') {
517
+ console.log('请求已中止');
518
+ // 在本地/全局状态中标记为已中止
519
+ setStreamingContent(prev => {
520
+ if (!prev) return prev;
521
+ const updated = { ...prev, is_aborted: true, is_streaming: false };
522
+ streamingContentRef.current = updated;
523
+ return updated;
524
+ });
525
+ } else if (err?.message === CHAT_STREAM_HANDLED_JSON_ERROR) {
526
+ // 已在 postChatStreamRequest 内 toast / 未登录跳转
527
+ } else {
528
+ console.error('流式错误:', err);
529
+ showToast(t('chat.send_failed') || '发送消息失败', 'error');
530
+ // 回滚用户消息?还是保留并显示错误状态?
531
+ // 当前逻辑保留消息。
532
+ }
533
+ } finally {
534
+ if (thisStreamGen !== streamGenerationRef.current) {
535
+ if (abortControllerRef.current === controller) {
536
+ abortControllerRef.current = null;
537
+ }
538
+ setStreamingPlaceholderThread(null);
539
+ return;
540
+ }
541
+
542
+ setIsStreaming(false);
543
+ if (abortControllerRef.current === controller) {
544
+ abortControllerRef.current = null;
545
+ }
546
+
547
+ // 在缓存中最终确定消息
548
+ if (streamingContentRef.current && streamingContentRef.current.content) {
549
+ const finalMsg = omitMessagePreparationForPersist(
550
+ streamingContentRef.current,
551
+ );
552
+ queryClient.setQueryData<Message[]>(['messages', threadId], (old) => {
553
+ const prev = old ?? [];
554
+ const exists = prev.some((m) => m.id === finalMsg.id);
555
+ if (!exists) return [...prev, finalMsg];
556
+ return prev.map((m) => (m.id === finalMsg.id ? finalMsg : m));
557
+ });
558
+ }
559
+
560
+ // 须先于 isFirstSessionTurn 内的 await:否则 index-session 会同时展示「缓存中的助手条 + streamingContent」
561
+ // 新会话首轮流式约 0.6s+ 的列表刷新窗口内会叠成两个回答
562
+ setStreamingContent(null);
563
+ streamingContentRef.current = null;
564
+
565
+ if (isFirstSessionTurn) {
566
+ const beforeThreads = queryClient.getQueryData<Thread[]>(['threads']) ?? [];
567
+ try {
568
+ await new Promise<void>((r) => {
569
+ window.setTimeout(r, THREADS_LIST_REFETCH_DELAY_MS);
570
+ });
571
+ await queryClient.refetchQueries({ queryKey: ['threads'] });
572
+ const afterThreads = queryClient.getQueryData<Thread[]>(['threads']) ?? [];
573
+ const afterIds = new Set(afterThreads.map((x) => x.id));
574
+ const orphan = beforeThreads.filter((x) => !afterIds.has(x.id));
575
+ if (orphan.length > 0) {
576
+ queryClient.setQueryData<Thread[]>(['threads'], [...orphan, ...afterThreads]);
577
+ }
578
+ } catch (e) {
579
+ console.error('[ChatContext] threads refetch failed', e);
580
+ queryClient.invalidateQueries({ queryKey: ['threads'] });
581
+ }
582
+ }
583
+ setStreamingPlaceholderThread(null);
584
+
585
+ /** 新会话首轮流式成功后:先 AI 生成标题,失败或空则用提问前 20 字更新 */
586
+ if (streamFinishedOk && isFirstSessionTurn) {
587
+ const questionTrimmed = message.trim()
588
+ const titleFromQuestion =
589
+ questionTrimmed.length > 20
590
+ ? questionTrimmed.slice(0, 20)
591
+ : questionTrimmed
592
+
593
+ void (async () => {
594
+ let applied = false
595
+ try {
596
+ const titleRes = await getChatTitleAPI({
597
+ sessionId: threadId,
598
+ })
599
+ const title = pickGeneratedChatTitle(titleRes)
600
+ if (title) {
601
+ await updateChatTitleAPI({
602
+ sessionId: threadId,
603
+ title,
604
+ })
605
+ applied = true
606
+ }
607
+ } catch (e) {
608
+ console.error(
609
+ '[ChatContext] getChatTitleAPI failed',
610
+ e,
611
+ )
612
+ }
613
+ if (!applied && titleFromQuestion) {
614
+ try {
615
+ await updateChatTitleAPI({
616
+ sessionId: threadId,
617
+ title: titleFromQuestion,
618
+ })
619
+ applied = true
620
+ } catch (e) {
621
+ console.error(
622
+ '[ChatContext] fallback title update failed',
623
+ e,
624
+ )
625
+ }
626
+ }
627
+ if (applied) {
628
+ await queryClient.invalidateQueries({
629
+ queryKey: ['threads'],
630
+ })
631
+ }
632
+ })();
633
+ }
634
+
635
+ streamingThreadIdRef.current = null;
636
+ setActiveThreadId(null);
637
+ }
638
+
639
+ }, [queryClient, showToast, t, isStreaming, hasPermission]);
640
+
641
+ const regenerateMessage = useCallback(async (
642
+ sessionId: string,
643
+ turnId: string,
644
+ options?: ChatStreamSendOptions,
645
+ ) => {
646
+ const tid = String(turnId ?? '').trim();
647
+ if (!sessionId || !tid) return;
648
+
649
+ if (isStreaming) {
650
+ console.warn('正在流式传输中,忽略重新生成请求');
651
+ return;
652
+ }
653
+
654
+ await queryClient.cancelQueries({ queryKey: ['messages', sessionId] });
655
+
656
+ const staleThreadId = streamingThreadIdRef.current;
657
+ const staleAssistant = streamingContentRef.current;
658
+
659
+ streamGenerationRef.current += 1;
660
+ const thisStreamGen = streamGenerationRef.current;
661
+
662
+ if (abortControllerRef.current) {
663
+ abortControllerRef.current.abort();
664
+ abortControllerRef.current = null;
665
+ if (
666
+ staleAssistant?.role === 'assistant' &&
667
+ staleAssistant.content &&
668
+ staleThreadId
669
+ ) {
670
+ const toMerge = omitMessagePreparationForPersist(
671
+ markMessagePreparationDone({
672
+ ...staleAssistant,
673
+ is_streaming: false,
674
+ }),
675
+ );
676
+ queryClient.setQueryData<Message[]>(
677
+ ['messages', staleThreadId],
678
+ (old) => {
679
+ const prev = old ?? [];
680
+ const exists = prev.some((m) => m.id === toMerge.id);
681
+ if (!exists) return [...prev, toMerge];
682
+ return prev.map((m) =>
683
+ m.id === toMerge.id ? toMerge : m,
684
+ );
685
+ },
686
+ );
687
+ }
688
+ }
689
+
690
+ streamingThreadIdRef.current = sessionId;
691
+
692
+ setActiveThreadId(sessionId);
693
+ setIsStreaming(true);
694
+ setStreamingContent(null);
695
+ streamingContentRef.current = null;
696
+
697
+ const controller = new AbortController();
698
+ abortControllerRef.current = controller;
699
+
700
+ let streamFinishedOk = false;
701
+
702
+ const initialStreamingMessage: Message = {
703
+ id: (typeof crypto.randomUUID === 'function')
704
+ ? `temp-${crypto.randomUUID()}`
705
+ : `temp-ai-stream-${Date.now()}`,
706
+ role: 'assistant',
707
+ content: '',
708
+ thoughts: [],
709
+ options: [],
710
+ preparation_items: [],
711
+ is_streaming: true,
712
+ created_at: new Date().toISOString(),
713
+ };
714
+
715
+ setStreamingContent(initialStreamingMessage);
716
+ streamingContentRef.current = initialStreamingMessage;
717
+
718
+ queryClient.setQueryData<Message[]>(['messages', sessionId], (old) => {
719
+ const prev = old ?? [];
720
+ const cleared = prev.map((m) =>
721
+ m.role === 'assistant' && m.options && m.options.length > 0
722
+ ? { ...m, options: [] as string[] }
723
+ : m,
724
+ );
725
+ return cleared.filter((m) => {
726
+ if (m.role !== 'assistant') return true;
727
+ return resolveMessageTurnIdForFeedback(m) !== tid;
728
+ });
729
+ });
730
+
731
+ try {
732
+ const streamFlags = clampChatStreamFeatureFlags(options, hasPermission);
733
+ const response = await postChatStreamRequest(
734
+ {
735
+ sessionId,
736
+ turnId: tid,
737
+ enableNetwork: streamFlags.enableNetwork,
738
+ enableKnowledgeBase: streamFlags.enableKnowledgeBase,
739
+ enableDataAnalysis: streamFlags.enableDataAnalysis,
740
+ },
741
+ { signal: controller.signal, regenerate: true },
742
+ );
743
+
744
+ await readSseStreamBody(response, {
745
+ thisStreamGen,
746
+ streamGenerationRef,
747
+ setStreamingContent,
748
+ setIsStreaming,
749
+ streamingContentRef,
750
+ });
751
+
752
+ streamFinishedOk = true;
753
+ } catch (err: any) {
754
+ if (thisStreamGen !== streamGenerationRef.current) {
755
+ // 已被新请求取代
756
+ } else if (err.name === 'AbortError') {
757
+ console.log('请求已中止');
758
+ setStreamingContent((prev) => {
759
+ if (!prev) return prev;
760
+ const updated = {
761
+ ...prev,
762
+ is_aborted: true,
763
+ is_streaming: false,
764
+ };
765
+ streamingContentRef.current = updated;
766
+ return updated;
767
+ });
768
+ } else if (err?.message === CHAT_STREAM_HANDLED_JSON_ERROR) {
769
+ void queryClient.invalidateQueries({
770
+ queryKey: ['messages', sessionId],
771
+ });
772
+ } else {
773
+ console.error('流式重新生成错误:', err);
774
+ showToast(t('chat.send_failed') || '发送消息失败', 'error');
775
+ void queryClient.invalidateQueries({
776
+ queryKey: ['messages', sessionId],
777
+ });
778
+ }
779
+ } finally {
780
+ if (thisStreamGen !== streamGenerationRef.current) {
781
+ if (abortControllerRef.current === controller) {
782
+ abortControllerRef.current = null;
783
+ }
784
+ return;
785
+ }
786
+
787
+ setIsStreaming(false);
788
+ if (abortControllerRef.current === controller) {
789
+ abortControllerRef.current = null;
790
+ }
791
+
792
+ if (
793
+ streamFinishedOk &&
794
+ streamingContentRef.current &&
795
+ streamingContentRef.current.content
796
+ ) {
797
+ const finalMsg = omitMessagePreparationForPersist(
798
+ streamingContentRef.current,
799
+ );
800
+ queryClient.setQueryData<Message[]>(['messages', sessionId], (old) => {
801
+ const prev = old ?? [];
802
+ const exists = prev.some((m) => m.id === finalMsg.id);
803
+ if (!exists) return [...prev, finalMsg];
804
+ return prev.map((m) => (m.id === finalMsg.id ? finalMsg : m));
805
+ });
806
+ }
807
+
808
+ bumpThreadToTopInCache(queryClient, sessionId);
809
+
810
+ setStreamingContent(null);
811
+ streamingContentRef.current = null;
812
+ streamingThreadIdRef.current = null;
813
+ setActiveThreadId(null);
814
+ }
815
+ }, [queryClient, showToast, t, isStreaming, hasPermission]);
816
+
817
+ return (
818
+ <ChatContext.Provider value={{
819
+ activeThreadId,
820
+ isStreaming,
821
+ streamingContent,
822
+ streamingPlaceholderThread,
823
+ sendMessage,
824
+ regenerateMessage,
825
+ stopStream,
826
+ resetStream
827
+ }}>
828
+ {children}
829
+ </ChatContext.Provider>
830
+ );
831
+ };
832
+
833
+ export const useChatContext = () => {
834
+ const context = useContext(ChatContext);
835
+ if (!context) {
836
+ throw new Error('useChatContext 必须在 ChatProvider 内使用');
837
+ }
838
+ return context;
839
+ };