update_label_relationships_logic.go 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. package label_relationship
  2. import (
  3. "context"
  4. "github.com/suyuan32/simple-admin-common/msg/errormsg"
  5. "wechat-api/ent"
  6. "wechat-api/ent/contact"
  7. "wechat-api/ent/custom_types"
  8. "wechat-api/ent/labelrelationship"
  9. "wechat-api/ent/messagerecords"
  10. "wechat-api/ent/soptask"
  11. "wechat-api/internal/svc"
  12. "wechat-api/internal/types"
  13. "wechat-api/internal/utils/dberrorhandler"
  14. "github.com/zeromicro/go-zero/core/logx"
  15. )
  16. type UpdateLabelRelationshipsLogic struct {
  17. logx.Logger
  18. ctx context.Context
  19. svcCtx *svc.ServiceContext
  20. }
  21. func NewUpdateLabelRelationshipsLogic(ctx context.Context, svcCtx *svc.ServiceContext) *UpdateLabelRelationshipsLogic {
  22. return &UpdateLabelRelationshipsLogic{
  23. Logger: logx.WithContext(ctx),
  24. ctx: ctx,
  25. svcCtx: svcCtx}
  26. }
  27. func (l *UpdateLabelRelationshipsLogic) UpdateLabelRelationships(req *types.LabelRelationshipsInfo) (resp *types.BaseMsgResp, err error) {
  28. // 开始事务
  29. tx, err := l.svcCtx.DB.Tx(context.Background())
  30. if err != nil {
  31. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  32. }
  33. // 获取联系人信息
  34. c, err := tx.Contact.Query().Where(contact.ID(*req.ContactId)).Only(l.ctx)
  35. // 获取联系人当前已关联的标签
  36. currentLabelRelationships, err := tx.LabelRelationship.Query().Where(labelrelationship.ContactID(*req.ContactId)).All(l.ctx)
  37. if err != nil {
  38. _ = tx.Rollback()
  39. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  40. }
  41. // 提取当前标签ID
  42. var currentLabelIds []uint64
  43. for _, relationship := range currentLabelRelationships {
  44. currentLabelIds = append(currentLabelIds, relationship.LabelID)
  45. }
  46. // 对比新旧标签ID,找出需要新增和移除的标签
  47. addLabelIds, removeLabelIds := compareLabelIds(req.LabelIds, currentLabelIds)
  48. // 如果 req.UpdateType 为空,或 req.UpdateType 的值为 “all” 时
  49. if req.UpdateType == nil || *req.UpdateType == "all" {
  50. // 删除需要移除的标签关系
  51. for _, id := range removeLabelIds {
  52. _, err := tx.LabelRelationship.
  53. Delete().
  54. Where(
  55. labelrelationship.ContactID(*req.ContactId),
  56. labelrelationship.LabelID(id),
  57. ).
  58. Exec(l.ctx)
  59. if err != nil {
  60. _ = tx.Rollback()
  61. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  62. }
  63. }
  64. }
  65. // 创建需要新增的标签关系
  66. for _, id := range addLabelIds {
  67. _, _ = tx.LabelRelationship.Create().
  68. SetLabelID(id).
  69. SetContactID(*req.ContactId).
  70. Save(l.ctx)
  71. //if err != nil {
  72. // return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  73. //}
  74. }
  75. // 获取所有 status 为 3 且 bot_wxid_list 包含 c.wx_wxid 的 sop_task
  76. sopTasks, err := tx.SopTask.Query().Where(soptask.Status(3)).All(l.ctx)
  77. if err != nil {
  78. _ = tx.Rollback()
  79. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  80. }
  81. var filteredSopTasks []*ent.SopTask
  82. for _, task := range sopTasks {
  83. for _, botWxid := range task.BotWxidList {
  84. if botWxid == c.WxWxid {
  85. filteredSopTasks = append(filteredSopTasks, task)
  86. break
  87. }
  88. }
  89. }
  90. // 获取所有 filteredSopTasks 的 sop_stages
  91. var sopStages []*ent.SopStage
  92. for _, task := range filteredSopTasks {
  93. stages, err := task.QueryTaskStages().All(l.ctx)
  94. if err != nil {
  95. _ = tx.Rollback()
  96. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  97. }
  98. sopStages = append(sopStages, stages...)
  99. }
  100. err = l.AddLabelRelationships(sopStages, *c, req.LabelIds)
  101. if err != nil {
  102. _ = tx.Rollback()
  103. return nil, err
  104. }
  105. // 所有操作成功,提交事务
  106. err = tx.Commit()
  107. if err != nil {
  108. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  109. }
  110. return &types.BaseMsgResp{Msg: errormsg.UpdateSuccess}, nil
  111. }
  112. func (l *UpdateLabelRelationshipsLogic) AddLabelRelationships(sopStages []*ent.SopStage, contact ent.Contact, currentLabelIds []uint64) (err error) {
  113. //// 开始事务
  114. //tx, err := l.svcCtx.DB.Tx(context.Background())
  115. //if err != nil {
  116. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  117. //}
  118. // 遍历 sop_stages,找出满足条件的 stage
  119. for _, stage := range sopStages {
  120. if stage.ConditionType == 1 && isLabelIdListMatchFilter(currentLabelIds, stage.ConditionOperator, stage.ConditionList) {
  121. // 判断是否有 contact_wxid、source_type、source_id、sub_source_id 相同的记录
  122. _, err = l.svcCtx.DB.MessageRecords.Query().
  123. Where(
  124. messagerecords.ContactWxid(contact.Wxid),
  125. messagerecords.SourceType(3),
  126. messagerecords.SourceID(stage.ID),
  127. messagerecords.SubSourceID(0),
  128. ).
  129. Only(l.ctx)
  130. if err == nil {
  131. continue
  132. }
  133. // 判断ActionMessage是否为空
  134. sourceType := 3
  135. if stage.ActionMessage != nil {
  136. for i, message := range stage.ActionMessage {
  137. meta := custom_types.Meta{}
  138. if message.Meta != nil {
  139. meta.Filename = message.Meta.Filename
  140. }
  141. _, _ = l.svcCtx.DB.MessageRecords.Create().
  142. SetNotNilBotWxid(&contact.WxWxid).
  143. SetNotNilContactID(&contact.ID).
  144. SetNotNilContactType(&contact.Type).
  145. SetNotNilContactWxid(&contact.Wxid).
  146. SetNotNilContentType(&message.Type).
  147. SetNotNilContent(&message.Content).
  148. SetMeta(meta).
  149. SetNotNilSourceType(&sourceType).
  150. SetNotNilSourceID(&stage.ID).
  151. SetSubSourceID(uint64(i)).
  152. Save(l.ctx)
  153. //if err != nil {
  154. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  155. //}
  156. }
  157. }
  158. if stage.ActionLabel != nil {
  159. // 获取 addLabelIds 中不在 currentLabelIds 中的标签ID
  160. var newLabelIds []uint64
  161. // 创建一个映射,用于快速查找 currentLabelIds 中的元素
  162. currentLabelIdSet := make(map[uint64]struct{})
  163. for _, id := range currentLabelIds {
  164. currentLabelIdSet[id] = struct{}{}
  165. }
  166. // 遍历 addLabelIds,找出不在 currentLabelIds 中的元素
  167. for _, id := range stage.ActionLabel {
  168. if _, exists := currentLabelIdSet[id]; !exists {
  169. newLabelIds = append(newLabelIds, id)
  170. }
  171. }
  172. if len(newLabelIds) == 0 {
  173. return nil
  174. }
  175. // 创建需要新增的标签关系
  176. for _, id := range newLabelIds {
  177. _, _ = l.svcCtx.DB.LabelRelationship.Create().
  178. SetLabelID(id).
  179. SetContactID(contact.ID).
  180. Save(l.ctx)
  181. //if err != nil {
  182. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  183. //}
  184. }
  185. // 合并 currentLabelIds 和 newLabelIds
  186. currentLabelIds = append(currentLabelIds, newLabelIds...)
  187. // 递归调用 AddLabelRelationships
  188. _ = l.AddLabelRelationships(sopStages, contact, currentLabelIds)
  189. return
  190. }
  191. }
  192. }
  193. // 所有操作成功,提交事务
  194. //err = tx.Commit()
  195. //if err != nil {
  196. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  197. //}
  198. return nil
  199. }
  200. // compareLabelIds compares the new label ids with the current ones and returns the ids to be added and removed
  201. func compareLabelIds(newLabelIds []uint64, currentLabelIds []uint64) (addLabelIds []uint64, removeLabelIds []uint64) {
  202. newLabelIdSet := make(map[uint64]struct{}, len(newLabelIds))
  203. for _, id := range newLabelIds {
  204. newLabelIdSet[id] = struct{}{}
  205. }
  206. for _, id := range currentLabelIds {
  207. if _, ok := newLabelIdSet[id]; ok {
  208. delete(newLabelIdSet, id)
  209. } else {
  210. removeLabelIds = append(removeLabelIds, id)
  211. }
  212. }
  213. for id := range newLabelIdSet {
  214. addLabelIds = append(addLabelIds, id)
  215. }
  216. return
  217. }
  218. func isLabelIdListMatchFilter(labelIdList []uint64, conditionOperator int, conditionList []custom_types.Condition) bool {
  219. labelIdSet := make(map[uint64]struct{})
  220. for _, id := range labelIdList {
  221. labelIdSet[id] = struct{}{}
  222. }
  223. for _, condition := range conditionList {
  224. match := false
  225. for _, id := range condition.LabelIdList {
  226. if _, ok := labelIdSet[id]; ok {
  227. match = true
  228. break
  229. }
  230. }
  231. if condition.Equal == 2 {
  232. match = !match
  233. }
  234. if (conditionOperator == 1 && !match) || (conditionOperator == 2 && match) {
  235. return match
  236. }
  237. }
  238. return conditionOperator == 1
  239. }