publish_sop_task_logic.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. package sop_task
  2. import (
  3. "context"
  4. "errors"
  5. "github.com/suyuan32/simple-admin-common/msg/errormsg"
  6. "regexp"
  7. "wechat-api/ent"
  8. "wechat-api/ent/contact"
  9. "wechat-api/ent/custom_types"
  10. "wechat-api/ent/labelrelationship"
  11. "wechat-api/ent/messagerecords"
  12. "wechat-api/ent/predicate"
  13. "wechat-api/ent/sopstage"
  14. "wechat-api/ent/soptask"
  15. "wechat-api/internal/utils/dberrorhandler"
  16. "wechat-api/internal/svc"
  17. "wechat-api/internal/types"
  18. "github.com/zeromicro/go-zero/core/logx"
  19. )
  20. type PublishSopTaskLogic struct {
  21. logx.Logger
  22. ctx context.Context
  23. svcCtx *svc.ServiceContext
  24. }
  25. func NewPublishSopTaskLogic(ctx context.Context, svcCtx *svc.ServiceContext) *PublishSopTaskLogic {
  26. return &PublishSopTaskLogic{
  27. Logger: logx.WithContext(ctx),
  28. ctx: ctx,
  29. svcCtx: svcCtx}
  30. }
  31. func (l *PublishSopTaskLogic) PublishSopTask(req *types.IDReq) (resp *types.BaseMsgResp, err error) {
  32. organizationId := l.ctx.Value("organizationId").(uint64)
  33. // 开始事务
  34. //tx, err := l.svcCtx.DB.Tx(context.Background())
  35. //if err != nil {
  36. // return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  37. //}
  38. // 根据 id 查询 sop_task
  39. sopTask, err := l.svcCtx.DB.SopTask.Query().
  40. Where(
  41. soptask.ID(req.Id),
  42. soptask.Status(1),
  43. soptask.OrganizationIDEQ(organizationId),
  44. ).
  45. WithTaskStages().
  46. Only(l.ctx)
  47. if err != nil {
  48. //_ = tx.Rollback()
  49. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  50. }
  51. // 判断 sop_task 是否存在
  52. if sopTask != nil {
  53. if sopTask.BotWxidList == nil {
  54. return nil, errors.New(errormsg.ValidationError)
  55. }
  56. // 查询任务的所有 sop_stages
  57. err = l.svcCtx.DB.SopTask.UpdateOneID(req.Id).
  58. SetStatus(3).
  59. Exec(l.ctx)
  60. if err != nil {
  61. //_ = tx.Rollback()
  62. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  63. }
  64. sopStages, err := l.svcCtx.DB.SopStage.Query().
  65. Where(sopstage.HasSopTaskWith(soptask.OrganizationIDEQ(organizationId), soptask.StatusEQ(3), soptask.DeletedAtIsNil()), sopstage.DeletedAtIsNil()).
  66. All(l.ctx)
  67. if err != nil {
  68. //_ = tx.Rollback()
  69. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  70. }
  71. stageMap := make(map[uint64]*ent.SopStage)
  72. for _, stage := range sopStages {
  73. stageMap[stage.ID] = stage
  74. }
  75. // 遍历 stage
  76. for _, stage := range sopTask.Edges.TaskStages {
  77. if stage.ConditionType == 1 {
  78. // 构造查询条件
  79. var predicates []predicate.Contact
  80. for _, condition := range stage.ConditionList {
  81. subPredicate := contact.HasContactRelationshipsWith(
  82. labelrelationship.LabelIDIn(condition.LabelIdList...))
  83. labelrelationship.OrganizationIDEQ(organizationId)
  84. if condition.Equal == 2 {
  85. subPredicate = contact.Not(subPredicate)
  86. }
  87. predicates = append(predicates, subPredicate)
  88. }
  89. // 查询满足条件的联系人
  90. var contacts []*ent.Contact
  91. var err error
  92. sourceType := 3
  93. if stage.ConditionOperator == 1 {
  94. contacts, err = l.svcCtx.DB.Contact.Query().Where(contact.And(predicates...)).All(l.ctx)
  95. } else {
  96. contacts, err = l.svcCtx.DB.Contact.Query().Where(contact.Or(predicates...)).All(l.ctx)
  97. }
  98. if err != nil {
  99. //_ = tx.Rollback()
  100. return nil, err
  101. }
  102. // 遍历 contacts
  103. for _, c := range contacts {
  104. // 判断联系人所属微信是否包含在任务当中
  105. if sopTask.BotWxidList == nil || (sopTask.BotWxidList != nil && valueInArray(c.WxWxid, sopTask.BotWxidList)) {
  106. if stage.ActionMessage != nil {
  107. for i, message := range stage.ActionMessage {
  108. meta := custom_types.Meta{}
  109. if message.Meta != nil {
  110. meta.Filename = message.Meta.Filename
  111. }
  112. _, _ = l.svcCtx.DB.MessageRecords.Create().
  113. SetNotNilBotWxid(&c.WxWxid).
  114. SetNotNilContactID(&c.ID).
  115. SetNotNilContactType(&c.Type).
  116. SetNotNilContactWxid(&c.Wxid).
  117. SetNotNilContentType(&message.Type).
  118. SetNotNilContent(&message.Content).
  119. SetMeta(meta).
  120. SetNotNilSourceType(&sourceType).
  121. SetNotNilSourceID(&stage.ID).
  122. SetSubSourceID(uint64(i)).
  123. SetOrganizationID(organizationId).
  124. Save(l.ctx)
  125. //if err != nil {
  126. // return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  127. //}
  128. }
  129. }
  130. if stage.ActionForward != nil {
  131. if stage.ActionForward.Wxid != "" {
  132. forwardWxids := splitString(stage.ActionForward.Wxid)
  133. for _, forwardWxid := range forwardWxids {
  134. for i, message := range stage.ActionForward.Action {
  135. meta := custom_types.Meta{}
  136. if message.Meta != nil {
  137. meta.Filename = message.Meta.Filename
  138. }
  139. _, err = l.svcCtx.DB.MessageRecords.Create().
  140. SetBotWxid(c.WxWxid).
  141. SetContactID(0).
  142. SetContactType(0).
  143. SetContactWxid(forwardWxid).
  144. SetContentType(message.Type).
  145. SetContent(message.Content).
  146. SetMeta(meta).
  147. SetSourceType(sourceType).
  148. SetSourceID(stage.ID).
  149. SetSubSourceID(c.ID + uint64(i)).
  150. SetOrganizationID(organizationId).
  151. Save(l.ctx)
  152. }
  153. }
  154. }
  155. }
  156. // 查询当前联系人的标签关系
  157. currentLabelRelationships, err := l.svcCtx.DB.LabelRelationship.Query().Where(labelrelationship.ContactID(c.ID)).All(l.ctx)
  158. if err != nil {
  159. //_ = tx.Rollback()
  160. return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  161. }
  162. // 提取当前标签ID
  163. var currentLabelIds []uint64
  164. for _, relationship := range currentLabelRelationships {
  165. currentLabelIds = append(currentLabelIds, relationship.LabelID)
  166. }
  167. if stage.ActionLabelAdd != nil || stage.ActionLabelDel != nil {
  168. // 递归调用 AddLabelRelationships
  169. err = l.AddLabelRelationships(stageMap, *c, currentLabelIds, stage.ActionLabelAdd, stage.ActionLabelDel, organizationId)
  170. if err != nil {
  171. //_ = tx.Rollback()
  172. return nil, err
  173. }
  174. }
  175. }
  176. }
  177. }
  178. }
  179. // 所有操作成功,提交事务
  180. //err = tx.Commit()
  181. //if err != nil {
  182. // return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  183. //}
  184. return &types.BaseMsgResp{Msg: errormsg.Success}, nil
  185. } else {
  186. // 所有操作成功,提交事务
  187. //err = tx.Commit()
  188. //if err != nil {
  189. // return nil, dberrorhandler.DefaultEntError(l.Logger, err, req)
  190. //}
  191. //// 返回错误信息:任务不存在
  192. return nil, errors.New(errormsg.TargetNotFound)
  193. }
  194. }
  195. func (l *PublishSopTaskLogic) AddLabelRelationships(sopStages map[uint64]*ent.SopStage, contact ent.Contact, currentLabelIds []uint64, addLabelIds []uint64, delLabelIds []uint64, organizationId uint64) (err error) {
  196. //// 开始事务
  197. //tx, err := l.svcCtx.DB.Tx(context.Background())
  198. //if err != nil {
  199. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  200. //}
  201. // 获取 addLabelIds 中不在 currentLabelIds 中的标签ID
  202. var newLabelIds []uint64
  203. var remLabelIds []uint64
  204. var finalLabelIds []uint64
  205. // 创建一个映射,用于快速查找 currentLabelIds 中的元素
  206. currentLabelIdSet := make(map[uint64]struct{})
  207. for _, id := range currentLabelIds {
  208. currentLabelIdSet[id] = struct{}{}
  209. }
  210. delLabelIdSet := make(map[uint64]struct{})
  211. for _, id := range delLabelIds {
  212. delLabelIdSet[id] = struct{}{}
  213. }
  214. if addLabelIds != nil && len(addLabelIds) > 0 {
  215. // 遍历 addLabelIds,找出不在 currentLabelIds 中的元素
  216. for _, id := range addLabelIds {
  217. if _, ce := currentLabelIdSet[id]; !ce {
  218. if _, re := delLabelIdSet[id]; !re {
  219. newLabelIds = append(newLabelIds, id)
  220. }
  221. }
  222. }
  223. if len(newLabelIds) > 0 {
  224. // 创建需要新增的标签关系
  225. for _, id := range newLabelIds {
  226. currentLabelIdSet[id] = struct{}{}
  227. _, err = l.svcCtx.DB.LabelRelationship.Create().
  228. SetLabelID(id).
  229. SetContactID(contact.ID).
  230. SetOrganizationID(organizationId).
  231. Save(l.ctx)
  232. if err != nil {
  233. //_ = tx.Rollback()
  234. return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  235. }
  236. }
  237. // 合并 currentLabelIds 和 newLabelIds
  238. currentLabelIds = append(currentLabelIds, newLabelIds...)
  239. }
  240. }
  241. if delLabelIds != nil && len(delLabelIds) > 0 {
  242. // 遍历 delLabelIds,找出在 currentLabelIds 中的元素
  243. for _, id := range delLabelIds {
  244. if _, exists := currentLabelIdSet[id]; exists {
  245. remLabelIds = append(remLabelIds, id)
  246. delete(currentLabelIdSet, id)
  247. }
  248. }
  249. if len(remLabelIds) > 0 {
  250. _, err = l.svcCtx.DB.LabelRelationship.Delete().Where(labelrelationship.LabelIDIn(remLabelIds...), labelrelationship.ContactIDEQ(contact.ID), labelrelationship.OrganizationIDEQ(organizationId)).Exec(l.ctx)
  251. if err != nil {
  252. //_ = tx.Rollback()
  253. return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  254. }
  255. }
  256. }
  257. if len(newLabelIds) == 0 && len(remLabelIds) == 0 {
  258. return nil
  259. }
  260. for id := range currentLabelIdSet {
  261. finalLabelIds = append(finalLabelIds, id)
  262. }
  263. // 遍历 sop_stages,找出满足条件的 stage
  264. for key, stage := range sopStages {
  265. if stage != nil && stage.ConditionType == 1 && isLabelIdListMatchFilter(finalLabelIds, stage.ConditionOperator, stage.ConditionList) {
  266. // 判断是否有 contact_wxid、source_type、source_id、sub_source_id 相同的记录
  267. _, err := l.svcCtx.DB.MessageRecords.Query().
  268. Where(
  269. messagerecords.ContactWxid(contact.Wxid),
  270. messagerecords.SourceType(3),
  271. messagerecords.SourceID(stage.ID),
  272. messagerecords.SubSourceID(0),
  273. ).
  274. Only(l.ctx)
  275. if !ent.IsNotFound(err) {
  276. continue
  277. }
  278. // 判断ActionMessage是否为空
  279. sourceType := 3
  280. if stage.ActionMessage != nil {
  281. for i, message := range stage.ActionMessage {
  282. meta := custom_types.Meta{}
  283. if message.Meta != nil {
  284. meta.Filename = message.Meta.Filename
  285. }
  286. _, _ = l.svcCtx.DB.MessageRecords.Create().
  287. SetNotNilBotWxid(&contact.WxWxid).
  288. SetNotNilContactID(&contact.ID).
  289. SetNotNilContactType(&contact.Type).
  290. SetNotNilContactWxid(&contact.Wxid).
  291. SetNotNilContentType(&message.Type).
  292. SetNotNilContent(&message.Content).
  293. SetMeta(meta).
  294. SetNotNilSourceType(&sourceType).
  295. SetNotNilSourceID(&stage.ID).
  296. SetSubSourceID(uint64(i)).
  297. SetOrganizationID(organizationId).
  298. Save(l.ctx)
  299. //if err != nil {
  300. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  301. //}
  302. }
  303. }
  304. if stage.ActionForward != nil {
  305. if stage.ActionForward.Wxid != "" {
  306. forwardWxids := splitString(stage.ActionForward.Wxid)
  307. for _, forwardWxid := range forwardWxids {
  308. for i, message := range stage.ActionForward.Action {
  309. meta := custom_types.Meta{}
  310. if message.Meta != nil {
  311. meta.Filename = message.Meta.Filename
  312. }
  313. _, err = l.svcCtx.DB.MessageRecords.Create().
  314. SetBotWxid(contact.WxWxid).
  315. SetContactID(0).
  316. SetContactType(0).
  317. SetContactWxid(forwardWxid).
  318. SetContentType(message.Type).
  319. SetContent(message.Content).
  320. SetMeta(meta).
  321. SetSourceType(sourceType).
  322. SetSourceID(stage.ID).
  323. SetSubSourceID(contact.ID + uint64(i)).
  324. SetOrganizationID(organizationId).
  325. Save(l.ctx)
  326. }
  327. }
  328. }
  329. }
  330. if stage.ActionLabelAdd != nil || stage.ActionLabelDel != nil {
  331. // 递归调用 AddLabelRelationships
  332. sopStages[key] = nil
  333. err = l.AddLabelRelationships(sopStages, contact, finalLabelIds, stage.ActionLabelAdd, stage.ActionLabelDel, organizationId)
  334. if err != nil {
  335. //_ = tx.Rollback()
  336. return err
  337. }
  338. }
  339. }
  340. }
  341. // 所有操作成功,提交事务
  342. //err = tx.Commit()
  343. //if err != nil {
  344. // return dberrorhandler.DefaultEntError(l.Logger, err, nil)
  345. //}
  346. return nil
  347. }
  348. func valueInArray(val string, array []string) bool {
  349. for _, item := range array {
  350. if item == val {
  351. return true
  352. }
  353. }
  354. return false
  355. }
  356. func isLabelIdListMatchFilter(labelIdList []uint64, conditionOperator int, conditionList []custom_types.Condition) bool {
  357. labelIdSet := make(map[uint64]struct{})
  358. for _, id := range labelIdList {
  359. labelIdSet[id] = struct{}{}
  360. }
  361. for _, condition := range conditionList {
  362. match := false
  363. for _, id := range condition.LabelIdList {
  364. if _, ok := labelIdSet[id]; ok {
  365. match = true
  366. break
  367. }
  368. }
  369. if condition.Equal == 2 {
  370. match = !match
  371. }
  372. if conditionOperator == 1 && !match {
  373. return false
  374. } else if conditionOperator == 2 && match {
  375. return true
  376. }
  377. }
  378. return conditionOperator == 1
  379. }
  380. func splitString(input string) []string {
  381. // Define the regular expression pattern to match Chinese comma, English comma, and Chinese enumeration comma
  382. pattern := `[,,、]`
  383. re := regexp.MustCompile(pattern)
  384. // Split the input string based on the pattern
  385. return re.Split(input, -1)
  386. }