wechat_platform.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  1. package llm
  2. import (
  3. "eta/eta_api/models"
  4. "eta/eta_api/models/rag"
  5. "eta/eta_api/utils"
  6. "eta/eta_api/utils/llm/eta_llm/eta_llm_http"
  7. "fmt"
  8. "html"
  9. "os"
  10. "regexp"
  11. "strconv"
  12. "strings"
  13. "time"
  14. )
  15. type WechatArticleOp struct {
  16. Source string
  17. WechatPlatformId int
  18. }
  19. // AddWechatPlatform
  20. // @Description: 添加新的公众号
  21. // @param item
  22. func AddWechatPlatform(item *rag.WechatPlatform) {
  23. var err error
  24. defer func() {
  25. if err != nil {
  26. utils.FileLog.Error("公众号入库后查找最新记录失败,err:%v", err)
  27. }
  28. }()
  29. if item.FakeId != `` {
  30. return
  31. }
  32. if item.ArticleLink == `` {
  33. return
  34. }
  35. articleLink := item.ArticleLink
  36. articleDetail, err := SearchByWechatArticle(item.ArticleLink)
  37. if err != nil {
  38. return
  39. }
  40. if articleDetail.Appuin == `` {
  41. err = fmt.Errorf("文章内未匹配到公众号唯一标识")
  42. return
  43. }
  44. wechatPlatform := new(rag.WechatPlatform)
  45. // 查找是否存在这个公众号id的
  46. wechatPlatformInfo, tmpErr := wechatPlatform.GetByFakeID(articleDetail.Appuin)
  47. if tmpErr != nil && !utils.IsErrNoRow(tmpErr) {
  48. err = tmpErr
  49. return
  50. }
  51. if tmpErr == nil {
  52. // 如果找到了,那么需要将当前的给移除掉
  53. err = item.Del()
  54. if err != nil {
  55. return
  56. }
  57. // 并将查出来的微信公众号摘出来的数据重新赋值
  58. item = wechatPlatformInfo
  59. } else if utils.IsErrNoRow(tmpErr) {
  60. // 如果没找到,那么就变更当前的信息
  61. item.FakeId = articleDetail.Appuin
  62. item.Nickname = articleDetail.Nickname
  63. //item.Alias = req.Alias
  64. item.RoundHeadImg = articleDetail.RoundHeadImg
  65. //item.ServiceType = req.ServiceType
  66. item.Signature = articleDetail.ProfileSignature
  67. //item.Verified = verified
  68. item.ModifyTime = time.Now()
  69. err = item.Update([]string{rag.WechatPlatformColumns.FakeID, rag.WechatPlatformColumns.Nickname, rag.WechatPlatformColumns.RoundHeadImg, rag.WechatPlatformColumns.Signature, rag.WechatPlatformColumns.ModifyTime})
  70. if err != nil {
  71. return
  72. }
  73. }
  74. // 把刚搜索的文章加入到指标库
  75. AddWechatArticle(item, articleLink, articleDetail, nil)
  76. BeachAddWechatArticle(item, 10)
  77. fmt.Println("公众号入库完成")
  78. return
  79. }
  80. // AddWechatArticle
  81. // @Description: 添加公众号文章入库
  82. // @author: Roc
  83. // @datetime 2025-03-05 13:24:14
  84. // @param item *rag.WechatPlatform
  85. // @param link string
  86. // @param articleDetail WechatArticleDataResp
  87. func AddWechatArticle(item *rag.WechatPlatform, articleLink string, articleDetail WechatArticleDataResp, articleMenu *ArticleMenu) {
  88. var err error
  89. defer func() {
  90. if err != nil {
  91. utils.FileLog.Error("公众号文章入库失败,文章链接:%s ,err:%v", articleLink, err)
  92. }
  93. }()
  94. obj := new(rag.WechatArticle)
  95. _, err = obj.GetByLink(articleLink)
  96. if err == nil {
  97. // 文章已经入库了,不需要重复入库
  98. return
  99. }
  100. // 如果不是 ErrNoRow 的时候,那么就是查询数据库出问题了,需要直接返回
  101. if !utils.IsErrNoRow(err) {
  102. return
  103. }
  104. // 这个时候,说明数据库中没有这个文章,那么需要文章入库
  105. err = nil
  106. var publishAt time.Time
  107. if articleDetail.CreateAt != `` {
  108. createAtInt, tmpErr := strconv.Atoi(articleDetail.CreateAt)
  109. if tmpErr == nil {
  110. publishAt = time.Unix(int64(createAtInt), 1000)
  111. }
  112. } else if articleMenu != nil {
  113. publishAt = time.Unix(int64(articleMenu.UpdateTime), 1000)
  114. }
  115. obj = &rag.WechatArticle{
  116. WechatArticleId: 0,
  117. WechatPlatformId: item.WechatPlatformId,
  118. FakeId: item.FakeId,
  119. Title: articleDetail.Title,
  120. Link: articleLink,
  121. CoverUrl: articleDetail.CoverUrl,
  122. Description: articleDetail.Desc,
  123. Content: html.EscapeString(articleDetail.HtmlContent),
  124. TextContent: articleDetail.TextContent,
  125. Country: articleDetail.CountryName,
  126. Province: articleDetail.ProvinceName,
  127. City: articleDetail.CityName,
  128. //Abstract: "",
  129. //ArticleCreateTime: createAt,
  130. ModifyTime: time.Now(),
  131. CreateTime: time.Now(),
  132. }
  133. if !publishAt.IsZero() {
  134. obj.ArticleCreateTime = publishAt
  135. }
  136. if articleMenu != nil {
  137. obj.Title = articleMenu.Title
  138. //obj.Link = articleMenu.Link
  139. obj.CoverUrl = articleMenu.Cover
  140. obj.Description = articleMenu.Digest
  141. }
  142. err = obj.Create()
  143. }
  144. // BeachAddWechatArticle
  145. // @Description: 批量添加公众号文章
  146. // @param item
  147. // @param num
  148. // @return err
  149. func BeachAddWechatArticle(item *rag.WechatPlatform, num int) {
  150. var err error
  151. defer func() {
  152. //fmt.Println("公众号文章批量入库完成")
  153. if err != nil {
  154. utils.FileLog.Error("公众号文章批量入库失败,err:%v", err)
  155. fmt.Println("公众号文章批量入库失败,err:", err)
  156. }
  157. }()
  158. if item.FakeId == `` {
  159. return
  160. }
  161. wechatArticleObj := new(rag.WechatArticle)
  162. // 获取公众号的文章列表
  163. articleListResp, err := SearchByWechatArticleList(item.FakeId, num)
  164. if err != nil {
  165. return
  166. }
  167. for _, articleMenu := range articleListResp.List {
  168. // 判断文章是否已经入库,如果已经入库了,那么就过滤,不去重复查询微信了
  169. _, err = wechatArticleObj.GetByLink(articleMenu.Link)
  170. if err == nil {
  171. // 文章已经入库了,不需要重复入库
  172. continue
  173. }
  174. if !utils.IsErrNoRow(err) {
  175. return
  176. }
  177. err = nil
  178. articleDetail, tmpErr := SearchByWechatArticle(articleMenu.Link)
  179. if tmpErr != nil {
  180. err = tmpErr
  181. return
  182. }
  183. // 把刚搜索的文章加入到指标库
  184. AddWechatArticle(item, articleMenu.Link, articleDetail, &articleMenu)
  185. time.Sleep(10 * time.Second)
  186. }
  187. return
  188. }
  189. // GenerateArticleAbstract
  190. // @Description: 文章摘要生成
  191. // @author: Roc
  192. // @datetime 2025-03-10 16:17:53
  193. // @param item *rag.WechatArticle
  194. func GenerateArticleAbstract(item *rag.WechatArticle) {
  195. var err error
  196. defer func() {
  197. if err != nil {
  198. utils.FileLog.Error("文章转临时文件失败,err:%v", err)
  199. fmt.Println("文章转临时文件失败,err:", err)
  200. }
  201. }()
  202. abstractObj := rag.WechatArticleAbstract{}
  203. _, err = abstractObj.GetByWechatArticleId(item.WechatArticleId)
  204. if err == nil {
  205. // 摘要已经生成,不需要重复生成
  206. return
  207. }
  208. if !utils.IsErrNoRow(err) {
  209. return
  210. }
  211. // 生成临时文件
  212. dateDir := time.Now().Format("20060102")
  213. uploadDir := utils.STATIC_DIR + "ai/" + dateDir
  214. err = os.MkdirAll(uploadDir, utils.DIR_MOD)
  215. if err != nil {
  216. err = fmt.Errorf("存储目录创建失败,Err:" + err.Error())
  217. return
  218. }
  219. randStr := utils.GetRandStringNoSpecialChar(28)
  220. fileName := randStr + `.md`
  221. tmpFilePath := uploadDir + "/" + fileName
  222. err = utils.SaveToFile(item.TextContent, tmpFilePath)
  223. if err != nil {
  224. err = fmt.Errorf("生成临时文件失败,Err:" + err.Error())
  225. return
  226. }
  227. defer func() {
  228. os.Remove(tmpFilePath)
  229. }()
  230. // 上传临时文件到LLM
  231. tmpFileResp, err := UploadTempDocs(tmpFilePath)
  232. if err != nil {
  233. err = fmt.Errorf("上传临时文件到LLM失败,Err:" + err.Error())
  234. return
  235. }
  236. if tmpFileResp.Data.Id == `` {
  237. err = fmt.Errorf("上传临时文件到LLM失败,Err:上传失败")
  238. return
  239. }
  240. tmpDocId := tmpFileResp.Data.Id
  241. //tmpDocId := `c4d2ee902808408c8b8ed398b33be103` // 钢材
  242. //tmpDocId := `2dde8afe62d24525a814e74e0a5e35e4` // 钢材
  243. //tmpDocId := `7634cc1086c04b3687682220a2cf1a48` //
  244. historyList := make([]eta_llm_http.HistoryContent, 0)
  245. questionObj := rag.Question{}
  246. questionList, err := questionObj.GetListByCondition(``, []interface{}{}, 0, 100)
  247. if err != nil {
  248. err = fmt.Errorf("获取问题列表失败,Err:" + err.Error())
  249. return
  250. }
  251. addArticleChatRecordList := make([]*rag.WechatArticleChatRecord, 0)
  252. var abstract string
  253. //开始对话
  254. for _, question := range questionList {
  255. originalAnswer, tmpAnswer, tmpErr := getAnswerByContent(tmpDocId, question.QuestionContent, historyList)
  256. if tmpErr != nil {
  257. err = fmt.Errorf("LLM对话失败,Err:" + tmpErr.Error())
  258. return
  259. }
  260. abstract = tmpAnswer
  261. historyList = append(historyList, eta_llm_http.HistoryContent{
  262. Role: `user`,
  263. Content: question.QuestionContent,
  264. }, eta_llm_http.HistoryContent{
  265. Role: `assistant`,
  266. Content: tmpAnswer,
  267. })
  268. // 待入库的数据
  269. addArticleChatRecordList = append(addArticleChatRecordList, &rag.WechatArticleChatRecord{
  270. WechatArticleChatRecordId: 0,
  271. WechatArticleId: item.WechatArticleId,
  272. ChatUserType: "user",
  273. Content: question.QuestionContent,
  274. SendTime: time.Now(),
  275. CreatedTime: time.Now(),
  276. UpdateTime: time.Now(),
  277. }, &rag.WechatArticleChatRecord{
  278. WechatArticleChatRecordId: 0,
  279. WechatArticleId: item.WechatArticleId,
  280. ChatUserType: "assistant",
  281. Content: originalAnswer,
  282. SendTime: time.Now(),
  283. CreatedTime: time.Now(),
  284. UpdateTime: time.Now(),
  285. })
  286. }
  287. // 添加问答记录
  288. if len(addArticleChatRecordList) > 0 {
  289. recordObj := rag.WechatArticleChatRecord{}
  290. err = recordObj.CreateInBatches(addArticleChatRecordList)
  291. if err != nil {
  292. return
  293. }
  294. }
  295. if abstract != `` {
  296. abstractItem := &rag.WechatArticleAbstract{
  297. WechatArticleAbstractId: 0,
  298. WechatArticleId: item.WechatArticleId,
  299. Content: abstract,
  300. Version: 0,
  301. VectorKey: "",
  302. ModifyTime: time.Now(),
  303. CreateTime: time.Now(),
  304. }
  305. err = abstractItem.Create()
  306. if err != nil {
  307. return
  308. }
  309. AbstractToKnowledge(item, abstractItem)
  310. }
  311. }
  312. func getAnswerByContent(docId, question string, historyList []eta_llm_http.HistoryContent) (originalAnswer, answer string, err error) {
  313. originalAnswer, result, err := ChatByFile(docId, question, historyList)
  314. fmt.Println(result)
  315. if err != nil {
  316. err = fmt.Errorf("LLM对话失败,Err:" + err.Error())
  317. return
  318. }
  319. // 提取 </think> 后面的内容
  320. thinkEndIndex := strings.Index(result.Answer, "</think>")
  321. if thinkEndIndex != -1 {
  322. answer = strings.TrimSpace(result.Answer[thinkEndIndex+len("</think>"):])
  323. } else {
  324. answer = result.Answer
  325. }
  326. answer = strings.TrimSpace(answer)
  327. return
  328. }
  329. // ArticleToKnowledge
  330. // @Description: 原文入向量库
  331. // @author: Roc
  332. // @datetime 2025-03-10 16:13:16
  333. // @param item *rag.WechatArticle
  334. func ArticleToKnowledge(item *rag.WechatArticle) {
  335. if item.TextContent == `` {
  336. return
  337. }
  338. var err error
  339. defer func() {
  340. if err != nil {
  341. utils.FileLog.Error("上传文章原文到知识库失败,err:%v", err)
  342. fmt.Println("上传文章原文到知识库失败,err:", err)
  343. }
  344. }()
  345. // 生成临时文件
  346. //dateDir := time.Now().Format("20060102")
  347. //uploadDir := utils.STATIC_DIR + "ai/article/" + dateDir
  348. uploadDir := utils.STATIC_DIR + "ai/article"
  349. err = os.MkdirAll(uploadDir, utils.DIR_MOD)
  350. if err != nil {
  351. err = fmt.Errorf("存储目录创建失败,Err:" + err.Error())
  352. return
  353. }
  354. fileName := RemoveSpecialChars(item.Title) + `.md`
  355. tmpFilePath := uploadDir + "/" + fileName
  356. err = utils.SaveToFile(item.TextContent, tmpFilePath)
  357. if err != nil {
  358. err = fmt.Errorf("生成临时文件失败,Err:" + err.Error())
  359. return
  360. }
  361. defer func() {
  362. os.Remove(tmpFilePath)
  363. }()
  364. knowledgeArticleName := models.BusinessConfMap[models.KnowledgeArticleName]
  365. // 上传临时文件到LLM
  366. uploadFileResp, err := UploadDocsToKnowledge(tmpFilePath, knowledgeArticleName)
  367. if err != nil {
  368. err = fmt.Errorf("上传文章原文到知识库失败,Err:" + err.Error())
  369. return
  370. }
  371. if len(uploadFileResp.FailedFiles) > 0 {
  372. for _, v := range uploadFileResp.FailedFiles {
  373. err = fmt.Errorf("上传文章原文到知识库失败,Err:" + v)
  374. }
  375. }
  376. item.VectorKey = tmpFilePath
  377. item.ModifyTime = time.Now()
  378. err = item.Update([]string{"vector_key", "modify_time"})
  379. }
  380. // AbstractToKnowledge
  381. // @Description: 摘要入向量库
  382. // @author: Roc
  383. // @datetime 2025-03-10 16:14:59
  384. // @param wechatArticleItem *rag.WechatArticle
  385. // @param item *rag.WechatArticleAbstract
  386. func AbstractToKnowledge(wechatArticleItem *rag.WechatArticle, item *rag.WechatArticleAbstract) {
  387. if item.Content == `` {
  388. return
  389. }
  390. // 已经生成了,那就不处理了
  391. if item.VectorKey != `` {
  392. return
  393. }
  394. var err error
  395. defer func() {
  396. if err != nil {
  397. utils.FileLog.Error("摘要入向量库失败,err:%v", err)
  398. fmt.Println("摘要入向量库失败,err:", err)
  399. }
  400. }()
  401. // 生成临时文件
  402. //dateDir := time.Now().Format("20060102")
  403. //uploadDir := utils.STATIC_DIR + "ai/article/" + dateDir
  404. uploadDir := utils.STATIC_DIR + "ai/abstract"
  405. err = os.MkdirAll(uploadDir, utils.DIR_MOD)
  406. if err != nil {
  407. err = fmt.Errorf("存储目录创建失败,Err:" + err.Error())
  408. return
  409. }
  410. fileName := RemoveSpecialChars(wechatArticleItem.Title) + `.md`
  411. tmpFilePath := uploadDir + "/" + fileName
  412. err = utils.SaveToFile(item.Content, tmpFilePath)
  413. if err != nil {
  414. err = fmt.Errorf("生成临时文件失败,Err:" + err.Error())
  415. return
  416. }
  417. defer func() {
  418. os.Remove(tmpFilePath)
  419. }()
  420. knowledgeArticleName := models.BusinessConfMap[models.KnowledgeBaseName]
  421. // 上传临时文件到LLM
  422. uploadFileResp, err := UploadDocsToKnowledge(tmpFilePath, knowledgeArticleName)
  423. if err != nil {
  424. err = fmt.Errorf("上传文章原文到知识库失败,Err:" + err.Error())
  425. return
  426. }
  427. if len(uploadFileResp.FailedFiles) > 0 {
  428. for _, v := range uploadFileResp.FailedFiles {
  429. err = fmt.Errorf("上传文章原文到知识库失败,Err:" + v)
  430. }
  431. }
  432. item.VectorKey = tmpFilePath
  433. item.ModifyTime = time.Now()
  434. err = item.Update([]string{"vector_key", "modify_time"})
  435. }
  436. func RemoveSpecialChars(text string) string {
  437. // 匹配非中文、非字母、非数字、非中文标点的字符
  438. reg := regexp.MustCompile(`[^\p{Han}\p{L}\p{N}\x{3000}-\x{303F}]`)
  439. return reg.ReplaceAllString(text, "")
  440. }