operation.go 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212
  1. package opsdev
  2. import (
  3. "context"
  4. "fmt"
  5. "regexp"
  6. "strings"
  7. "dashoo.cn/common_definition/comm_def"
  8. "dashoo.cn/opms_libary/myerrors"
  9. opsdevdao "dashoo.cn/opms_parent/app/dao/opsdev"
  10. opsdevmodel "dashoo.cn/opms_parent/app/model/opsdev"
  11. "dashoo.cn/opms_parent/app/service"
  12. "github.com/gogf/gf/database/gdb"
  13. "github.com/gogf/gf/frame/g"
  14. "github.com/gogf/gf/os/gtime"
  15. "github.com/gogf/gf/util/gconv"
  16. "github.com/gogf/gf/util/gvalid"
  17. )
  18. type OperationService struct {
  19. *service.ContextService
  20. Dao *opsdevdao.OpsOperationEventDao
  21. RecordDao *opsdevdao.OpsOperationEventRecordDao
  22. AttachmentDao *opsdevdao.OpsOperationEventAttachmentDao
  23. DeliveryProjectAttachmentDao *opsdevdao.OpsDeliveryProjectEventAttachmentDao
  24. }
  25. func NewOperationService(ctx context.Context) (svc *OperationService, err error) {
  26. svc = new(OperationService)
  27. if svc.ContextService, err = svc.Init(ctx); err != nil {
  28. return nil, err
  29. }
  30. svc.Dao = opsdevdao.NewOpsOperationEventDao(svc.Tenant)
  31. svc.RecordDao = opsdevdao.NewOpsOperationEventRecordDao(svc.Tenant)
  32. svc.AttachmentDao = opsdevdao.NewOpsOperationEventAttachmentDao(svc.Tenant)
  33. svc.DeliveryProjectAttachmentDao = opsdevdao.NewOpsDeliveryProjectEventAttachmentDao(svc.Tenant)
  34. return svc, nil
  35. }
  36. // GetList 运维事件列表
  37. func (s *OperationService) GetList(req *opsdevmodel.OpsOperationEventSearchReq) (total int, list []*opsdevmodel.OpsOperationEvent, err error) {
  38. db := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).Where(s.Dao.Columns.EventStatus+" != ?", opsdevmodel.EventStatusClosed)
  39. if req.EventNo != "" {
  40. db = db.Where(s.Dao.Columns.EventNo, req.EventNo)
  41. }
  42. if req.EventTitle != "" {
  43. db = db.Where(s.Dao.Columns.EventTitle+" like ?", "%"+req.EventTitle+"%")
  44. }
  45. if req.EventStatus != "" {
  46. db = db.Where(s.Dao.Columns.EventStatus, req.EventStatus)
  47. }
  48. if req.EventType != "" {
  49. db = db.Where(s.Dao.Columns.EventType, req.EventType)
  50. }
  51. if req.PriorityLevel != "" {
  52. db = db.Where(s.Dao.Columns.PriorityLevel, req.PriorityLevel)
  53. }
  54. if req.IsBig != "" {
  55. db = db.Where(s.Dao.Columns.IsBig, req.IsBig)
  56. }
  57. if req.IsOps != "" {
  58. db = db.Where(s.Dao.Columns.IsOps, req.IsOps)
  59. }
  60. if req.CustName != "" {
  61. db = db.Where(s.Dao.Columns.CustName+" like ?", "%"+req.CustName+"%")
  62. }
  63. if req.OpsUserName != "" {
  64. db = db.Where(s.Dao.Columns.OpsUserName+" like ?", "%"+req.OpsUserName+"%")
  65. }
  66. if req.ContractName != "" {
  67. db = db.Where(s.Dao.Columns.ContractName+" like ?", "%"+req.ContractName+"%")
  68. }
  69. if req.ProductLine != "" {
  70. db = db.Where(s.Dao.Columns.ProductLine, req.ProductLine)
  71. }
  72. if req.BeginTime != "" {
  73. db = db.Where(s.Dao.Columns.CreatedTime+" >= ?", req.BeginTime)
  74. }
  75. if req.EndTime != "" {
  76. db = db.Where(s.Dao.Columns.CreatedTime+" <= ?", req.EndTime)
  77. }
  78. total, err = db.Count()
  79. if err != nil {
  80. g.Log().Error(err)
  81. return 0, nil, myerrors.DbError("查询运维事件数量失败")
  82. }
  83. pageNum, pageSize := req.GetPage()
  84. err = db.Page(pageNum, pageSize).Order(s.Dao.Columns.CreatedTime + " desc").Scan(&list)
  85. if err != nil {
  86. g.Log().Error(err)
  87. return 0, nil, myerrors.DbError("查询运维事件列表失败")
  88. }
  89. return total, list, nil
  90. }
  91. // GetEntityById 运维事件详情
  92. func (s *OperationService) GetEntityById(req *comm_def.IdReq) (detail *opsdevmodel.OpsOperationEvent, err error) {
  93. if req.Id <= 0 {
  94. return nil, myerrors.ValidError("参数有误!")
  95. }
  96. detail = new(opsdevmodel.OpsOperationEvent)
  97. err = s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.Id).Scan(detail)
  98. if err != nil {
  99. g.Log().Error(err)
  100. return nil, myerrors.DbError("查询运维事件详情失败")
  101. }
  102. if detail.Id <= 0 {
  103. return nil, myerrors.TipsError("运维事件不存在")
  104. }
  105. return detail, nil
  106. }
  107. // Create 创建运维事件
  108. func (s *OperationService) Create(req *opsdevmodel.OpsOperationEventReq) (err error) {
  109. if err = gvalid.CheckStruct(s.Ctx, req, nil); err != nil {
  110. return myerrors.ValidError(err.Error())
  111. }
  112. data := gconv.Map(req)
  113. service.SetCreatedInfo(data, s.GetCxtUserId(), s.GetCxtUserName())
  114. data["eventNo"] = generateEventNo()
  115. data["eventStatus"] = opsdevmodel.EventStatusProcessing
  116. data["opsUserId"] = s.GetCxtUserId()
  117. data["opsUserName"] = s.GetCxtUserName()
  118. if v, ok := data["feedbackDate"]; ok && v == "" {
  119. data["feedbackDate"] = gtime.Now()
  120. }
  121. if v, ok := data["assignTime"]; ok && v == "" {
  122. data["assignTime"] = gtime.Now()
  123. }
  124. delete(data, "completeTime")
  125. delete(data, "attachments")
  126. result, err := s.Dao.Data(data).Insert()
  127. if err != nil {
  128. g.Log().Error(err)
  129. return myerrors.DbError("创建运维事件失败")
  130. }
  131. id, err := result.LastInsertId()
  132. if err != nil {
  133. g.Log().Error(err)
  134. return nil
  135. }
  136. recordData := g.Map{
  137. s.RecordDao.Columns.EventId: id,
  138. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  139. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  140. s.RecordDao.Columns.HandleContent: "创建运维事件<br/>说明: 事件已创建",
  141. s.RecordDao.Columns.HandleResult: opsdevmodel.HandleResultUnresolved,
  142. s.RecordDao.Columns.HandleDate: gtime.Now(),
  143. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeProcess,
  144. }
  145. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  146. _, err = s.RecordDao.Data(recordData).Insert()
  147. if err != nil {
  148. g.Log().Error(err)
  149. }
  150. // 保存附件到交付项目事件附件表
  151. if len(req.Attachments) > 0 {
  152. for _, attachment := range req.Attachments {
  153. attachData := g.Map{
  154. s.DeliveryProjectAttachmentDao.Columns.EventId: id,
  155. s.DeliveryProjectAttachmentDao.Columns.FileName: attachment.FileName,
  156. s.DeliveryProjectAttachmentDao.Columns.FileUrl: attachment.FileUrl,
  157. s.DeliveryProjectAttachmentDao.Columns.FileType: attachment.FileType,
  158. }
  159. service.SetCreatedInfo(attachData, s.GetCxtUserId(), s.GetCxtUserName())
  160. _, err := s.DeliveryProjectAttachmentDao.Data(attachData).Insert()
  161. if err != nil {
  162. g.Log().Error(err)
  163. }
  164. }
  165. }
  166. return nil
  167. }
  168. // UpdateById 更新运维事件
  169. func (s *OperationService) UpdateById(req *opsdevmodel.UpdateOpsOperationEventReq) (err error) {
  170. if err = gvalid.CheckStruct(s.Ctx, req, nil); err != nil {
  171. return myerrors.ValidError(err.Error())
  172. }
  173. event := new(opsdevmodel.OpsOperationEvent)
  174. err = s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.Id).Scan(event)
  175. if err != nil {
  176. g.Log().Error(err)
  177. return myerrors.DbError("查询运维事件失败")
  178. }
  179. if event.Id <= 0 {
  180. return myerrors.TipsError("运维事件不存在")
  181. }
  182. if err = s.checkNotClosed(event); err != nil {
  183. return err
  184. }
  185. data := gconv.Map(req.OpsOperationEventReq)
  186. if v, ok := data["feedbackDate"]; ok && v == "" {
  187. delete(data, "feedbackDate")
  188. }
  189. delete(data, "assignTime")
  190. delete(data, "completeTime")
  191. delete(data, "attachments")
  192. service.SetUpdatedInfo(data, s.GetCxtUserId(), s.GetCxtUserName())
  193. _, err = s.Dao.FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, req.Id).Data(data).Update()
  194. if err != nil {
  195. g.Log().Error(err)
  196. return myerrors.DbError("更新运维事件失败")
  197. }
  198. // 保存附件到交付项目事件附件表
  199. if len(req.Attachments) > 0 {
  200. for _, attachment := range req.Attachments {
  201. attachData := g.Map{
  202. s.DeliveryProjectAttachmentDao.Columns.EventId: req.Id,
  203. s.DeliveryProjectAttachmentDao.Columns.FileName: attachment.FileName,
  204. s.DeliveryProjectAttachmentDao.Columns.FileUrl: attachment.FileUrl,
  205. s.DeliveryProjectAttachmentDao.Columns.FileType: attachment.FileType,
  206. }
  207. service.SetCreatedInfo(attachData, s.GetCxtUserId(), s.GetCxtUserName())
  208. _, err := s.DeliveryProjectAttachmentDao.Data(attachData).Insert()
  209. if err != nil {
  210. g.Log().Error(err)
  211. }
  212. }
  213. }
  214. return nil
  215. }
  216. // DeleteByIds 批量删除运维事件
  217. func (s *OperationService) DeleteByIds(req *comm_def.IdsReq) (err error) {
  218. if len(req.Ids) == 0 {
  219. return myerrors.ValidError("参数有误!")
  220. }
  221. _, err = s.Dao.WherePri(req.Ids).Data(g.Map{
  222. s.Dao.Columns.DeletedTime: gtime.Now(),
  223. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  224. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  225. }).Update()
  226. if err != nil {
  227. g.Log().Error(err)
  228. return myerrors.DbError("删除运维事件失败")
  229. }
  230. return nil
  231. }
  232. // checkNotClosed 校验事件未关闭,已关闭的事件不允许操作
  233. func (s *OperationService) checkNotClosed(event *opsdevmodel.OpsOperationEvent) error {
  234. if event.EventStatus == opsdevmodel.EventStatusClosed {
  235. return myerrors.TipsError("该事件已关闭,无法进行操作")
  236. }
  237. return nil
  238. }
  239. // Process 处理运维事件
  240. func (s *OperationService) Process(ctx context.Context, req *opsdevmodel.OpsOperationEventProcessReq) error {
  241. if req.Id <= 0 {
  242. return myerrors.TipsError("任务ID不能为空")
  243. }
  244. event := new(opsdevmodel.OpsOperationEvent)
  245. err := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.Id).Scan(event)
  246. if err != nil {
  247. g.Log().Error(err)
  248. return myerrors.DbError("查询运维事件失败")
  249. }
  250. if event.Id <= 0 {
  251. return myerrors.TipsError("运维事件不存在")
  252. }
  253. if err := s.checkNotClosed(event); err != nil {
  254. return err
  255. }
  256. switch req.OperateType {
  257. case opsdevmodel.OperateTypeProcess:
  258. return s.processStart(ctx, event, req)
  259. case opsdevmodel.OperateTypeResume:
  260. return s.processResume(ctx, event, req)
  261. case opsdevmodel.OperateTypeTransfer:
  262. return s.processTransfer(ctx, event, req)
  263. case opsdevmodel.OperateTypeSuspend:
  264. return s.processSuspend(ctx, event, req)
  265. case opsdevmodel.OperateTypeClose:
  266. return s.processClose(ctx, event, req)
  267. default:
  268. return myerrors.TipsError("未知的操作类型")
  269. }
  270. }
  271. func (s *OperationService) processStart(ctx context.Context, event *opsdevmodel.OpsOperationEvent, req *opsdevmodel.OpsOperationEventProcessReq) error {
  272. if event.EventStatus != opsdevmodel.EventStatusPending {
  273. return myerrors.TipsError("当前事件状态不能进行此操作")
  274. }
  275. return s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  276. _, err := s.Dao.TX(tx).FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, event.Id).Data(g.Map{
  277. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusProcessing,
  278. s.Dao.Columns.OpsUserId: s.GetCxtUserId(),
  279. s.Dao.Columns.OpsUserName: s.GetCxtUserName(),
  280. s.Dao.Columns.AssignTime: gtime.Now(),
  281. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  282. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  283. s.Dao.Columns.UpdatedTime: gtime.Now(),
  284. }).Update()
  285. if err != nil {
  286. return err
  287. }
  288. recordData := g.Map{
  289. s.RecordDao.Columns.EventId: event.Id,
  290. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  291. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  292. s.RecordDao.Columns.HandleContent: req.HandleContent,
  293. s.RecordDao.Columns.HandleResult: opsdevmodel.HandleResultUnresolved,
  294. s.RecordDao.Columns.HandleDate: gtime.Now(),
  295. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeProcess,
  296. }
  297. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  298. _, err = s.RecordDao.TX(tx).Data(recordData).Insert()
  299. if err != nil {
  300. return err
  301. }
  302. return nil
  303. })
  304. }
  305. func (s *OperationService) processClose(ctx context.Context, event *opsdevmodel.OpsOperationEvent, req *opsdevmodel.OpsOperationEventProcessReq) error {
  306. if event.EventStatus == opsdevmodel.EventStatusPending {
  307. return myerrors.TipsError("待处理状态不能直接关闭,请先接收")
  308. }
  309. if event.EventStatus == opsdevmodel.EventStatusSuspended {
  310. return myerrors.TipsError("挂起状态不允许关闭事件,请先转处理")
  311. }
  312. return s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  313. _, err := s.Dao.TX(tx).FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, event.Id).Data(g.Map{
  314. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusClosed,
  315. s.Dao.Columns.CompleteTime: gtime.Now(),
  316. s.Dao.Columns.CompleteDesc: req.HandleContent,
  317. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  318. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  319. s.Dao.Columns.UpdatedTime: gtime.Now(),
  320. }).Update()
  321. if err != nil {
  322. return err
  323. }
  324. recordData := g.Map{
  325. s.RecordDao.Columns.EventId: event.Id,
  326. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  327. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  328. s.RecordDao.Columns.HandleContent: req.HandleContent,
  329. s.RecordDao.Columns.HandleResult: req.HandleResult,
  330. s.RecordDao.Columns.HandleDate: gtime.Now(),
  331. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeClose,
  332. }
  333. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  334. _, err = s.RecordDao.TX(tx).Data(recordData).Insert()
  335. if err != nil {
  336. return err
  337. }
  338. return nil
  339. })
  340. }
  341. func (s *OperationService) processTransfer(ctx context.Context, event *opsdevmodel.OpsOperationEvent, req *opsdevmodel.OpsOperationEventProcessReq) error {
  342. if event.EventStatus != opsdevmodel.EventStatusProcessing && event.EventStatus != opsdevmodel.EventStatusProcessingNormal {
  343. return myerrors.TipsError("当前事件状态不能转研发")
  344. }
  345. return s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  346. _, err := s.Dao.TX(tx).FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, event.Id).Data(g.Map{
  347. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusTransfer,
  348. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  349. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  350. s.Dao.Columns.UpdatedTime: gtime.Now(),
  351. }).Update()
  352. if err != nil {
  353. return err
  354. }
  355. recordData := g.Map{
  356. s.RecordDao.Columns.EventId: event.Id,
  357. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  358. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  359. s.RecordDao.Columns.HandleContent: req.HandleContent,
  360. s.RecordDao.Columns.HandleResult: opsdevmodel.HandleResultUnresolved,
  361. s.RecordDao.Columns.HandleDate: gtime.Now(),
  362. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeTransfer,
  363. }
  364. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  365. _, err = s.RecordDao.TX(tx).Data(recordData).Insert()
  366. if err != nil {
  367. return err
  368. }
  369. return nil
  370. })
  371. }
  372. func (s *OperationService) processSuspend(ctx context.Context, event *opsdevmodel.OpsOperationEvent, req *opsdevmodel.OpsOperationEventProcessReq) error {
  373. if event.EventStatus != opsdevmodel.EventStatusProcessing && event.EventStatus != opsdevmodel.EventStatusProcessingNormal && event.EventStatus != opsdevmodel.EventStatusTransfer {
  374. return myerrors.TipsError("当前事件状态不能挂起")
  375. }
  376. return s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  377. _, err := s.Dao.TX(tx).FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, event.Id).Data(g.Map{
  378. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusSuspended,
  379. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  380. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  381. s.Dao.Columns.UpdatedTime: gtime.Now(),
  382. }).Update()
  383. if err != nil {
  384. return err
  385. }
  386. recordData := g.Map{
  387. s.RecordDao.Columns.EventId: event.Id,
  388. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  389. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  390. s.RecordDao.Columns.HandleContent: req.HandleContent,
  391. s.RecordDao.Columns.HandleResult: opsdevmodel.HandleResultUnresolved,
  392. s.RecordDao.Columns.HandleDate: gtime.Now(),
  393. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeSuspend,
  394. }
  395. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  396. _, err = s.RecordDao.TX(tx).Data(recordData).Insert()
  397. if err != nil {
  398. return err
  399. }
  400. return nil
  401. })
  402. }
  403. func (s *OperationService) processResume(ctx context.Context, event *opsdevmodel.OpsOperationEvent, req *opsdevmodel.OpsOperationEventProcessReq) error {
  404. if event.EventStatus != opsdevmodel.EventStatusSuspended {
  405. return myerrors.TipsError("当前事件状态不能转处理")
  406. }
  407. return s.Dao.Transaction(context.TODO(), func(ctx context.Context, tx *gdb.TX) error {
  408. _, err := s.Dao.TX(tx).FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, event.Id).Data(g.Map{
  409. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusProcessing,
  410. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  411. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  412. s.Dao.Columns.UpdatedTime: gtime.Now(),
  413. }).Update()
  414. if err != nil {
  415. return err
  416. }
  417. recordData := g.Map{
  418. s.RecordDao.Columns.EventId: event.Id,
  419. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  420. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  421. s.RecordDao.Columns.HandleContent: req.HandleContent,
  422. s.RecordDao.Columns.HandleResult: opsdevmodel.HandleResultUnresolved,
  423. s.RecordDao.Columns.HandleDate: gtime.Now(),
  424. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeResume,
  425. }
  426. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  427. _, err = s.RecordDao.TX(tx).Data(recordData).Insert()
  428. if err != nil {
  429. return err
  430. }
  431. return nil
  432. })
  433. }
  434. // GetRecords 获取运维事件处理记录
  435. func (s *OperationService) GetRecords(req *opsdevmodel.OpsOperationEventRecordSearchReq) (total int, list []*opsdevmodel.OpsOperationEventRecordWithAttachments, err error) {
  436. if req.EventId <= 0 {
  437. return 0, nil, myerrors.ValidError("事件ID不能为空")
  438. }
  439. db := s.RecordDao.FieldsEx(s.RecordDao.Columns.DeletedTime).Where(s.RecordDao.Columns.EventId, req.EventId)
  440. total, err = db.Count()
  441. if err != nil {
  442. g.Log().Error(err)
  443. return 0, nil, myerrors.DbError("查询处理记录数量失败")
  444. }
  445. var records []*opsdevmodel.OpsOperationEventRecord
  446. pageNum, pageSize := req.GetPage()
  447. err = db.Page(pageNum, pageSize).Order(s.RecordDao.Columns.CreatedTime + " desc").Scan(&records)
  448. if err != nil {
  449. g.Log().Error(err)
  450. return 0, nil, myerrors.DbError("查询处理记录列表失败")
  451. }
  452. // 查询每个记录的附件
  453. list = make([]*opsdevmodel.OpsOperationEventRecordWithAttachments, 0, len(records))
  454. for _, record := range records {
  455. recordWithAttachments := &opsdevmodel.OpsOperationEventRecordWithAttachments{
  456. OpsOperationEventRecord: *record,
  457. Attachments: make([]*opsdevmodel.OpsOperationEventAttachment, 0),
  458. }
  459. // 查询该记录的附件
  460. var attachments []*opsdevmodel.OpsOperationEventAttachment
  461. err = s.AttachmentDao.FieldsEx(s.AttachmentDao.Columns.DeletedTime).
  462. Where(s.AttachmentDao.Columns.EventRecordId, record.Id).
  463. Order(s.AttachmentDao.Columns.CreatedTime + " desc").
  464. Scan(&attachments)
  465. if err != nil {
  466. g.Log().Error(err)
  467. } else {
  468. recordWithAttachments.Attachments = attachments
  469. }
  470. list = append(list, recordWithAttachments)
  471. }
  472. return total, list, nil
  473. }
  474. // UploadAttachment 上传附件
  475. func (s *OperationService) UploadAttachment(req *opsdevmodel.OpsOperationEventAttachmentReq) (attachment *opsdevmodel.OpsOperationEventAttachment, err error) {
  476. if req.EventId <= 0 {
  477. return nil, myerrors.ValidError("事件ID不能为空")
  478. }
  479. if req.FileName == "" {
  480. return nil, myerrors.ValidError("文件名不能为空")
  481. }
  482. if req.FileUrl == "" {
  483. return nil, myerrors.ValidError("文件地址不能为空")
  484. }
  485. event := new(opsdevmodel.OpsOperationEvent)
  486. err = s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.EventId).Scan(event)
  487. if err != nil {
  488. g.Log().Error(err)
  489. return nil, myerrors.DbError("查询运维事件失败")
  490. }
  491. if err := s.checkNotClosed(event); err != nil {
  492. return nil, err
  493. }
  494. data := g.Map{
  495. s.AttachmentDao.Columns.EventId: req.EventId,
  496. s.AttachmentDao.Columns.EventRecordId: req.EventRecordId,
  497. s.AttachmentDao.Columns.FileName: req.FileName,
  498. s.AttachmentDao.Columns.FileUrl: req.FileUrl,
  499. s.AttachmentDao.Columns.FileType: req.FileType,
  500. s.AttachmentDao.Columns.Remark: req.Remark,
  501. }
  502. service.SetCreatedInfo(data, s.GetCxtUserId(), s.GetCxtUserName())
  503. _, err = s.AttachmentDao.Data(data).Insert()
  504. if err != nil {
  505. g.Log().Error(err)
  506. return nil, myerrors.DbError("上传附件失败")
  507. }
  508. // 查询刚插入的记录
  509. attachment = new(opsdevmodel.OpsOperationEventAttachment)
  510. err = s.AttachmentDao.Where(s.AttachmentDao.Columns.EventId, req.EventId).
  511. Order(s.AttachmentDao.Columns.Id + " desc").Scan(attachment)
  512. if err != nil {
  513. g.Log().Error(err)
  514. return nil, myerrors.DbError("查询附件失败")
  515. }
  516. return attachment, nil
  517. }
  518. // GetAttachments 获取事件附件列表
  519. func (s *OperationService) GetAttachments(eventId int) (list []*opsdevmodel.OpsOperationEventAttachment, err error) {
  520. if eventId <= 0 {
  521. return nil, myerrors.ValidError("事件ID不能为空")
  522. }
  523. err = s.AttachmentDao.FieldsEx(s.AttachmentDao.Columns.DeletedTime).
  524. Where(s.AttachmentDao.Columns.EventId, eventId).
  525. Order(s.AttachmentDao.Columns.CreatedTime + " desc").Scan(&list)
  526. if err != nil {
  527. g.Log().Error(err)
  528. return nil, myerrors.DbError("查询附件列表失败")
  529. }
  530. return list, nil
  531. }
  532. // GetStats 运维事件统计
  533. func (s *OperationService) GetStats() (result g.Map, err error) {
  534. result = g.Map{}
  535. statusCounts := []struct {
  536. key string
  537. status string
  538. }{
  539. {"pending", opsdevmodel.EventStatusPending},
  540. {"processingKey", opsdevmodel.EventStatusProcessing},
  541. {"processingNormal", opsdevmodel.EventStatusProcessingNormal},
  542. {"transfer", opsdevmodel.EventStatusTransfer},
  543. {"suspended", opsdevmodel.EventStatusSuspended},
  544. {"closed", opsdevmodel.EventStatusClosed},
  545. }
  546. for _, sc := range statusCounts {
  547. count, err := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).Where(s.Dao.Columns.EventStatus, sc.status).Count()
  548. if err != nil {
  549. g.Log().Error(err)
  550. return nil, myerrors.DbError("查询运维事件统计失败")
  551. }
  552. result[sc.key] = count
  553. }
  554. today := gtime.Now().Format("Y-m-d")
  555. todayNewCount, err := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).Where(s.Dao.Columns.CreatedTime+" >= ?", today+" 00:00:00").Count()
  556. if err != nil {
  557. g.Log().Error(err)
  558. return nil, myerrors.DbError("查询今日新增数量失败")
  559. }
  560. result["todayNew"] = todayNewCount
  561. return result, nil
  562. }
  563. // generateEventNo 生成事件编号
  564. func generateEventNo() string {
  565. return fmt.Sprintf("OPS%s", gtime.Now().Format("YmdHis"))
  566. }
  567. // GetKanbanData 看板数据
  568. func (s *OperationService) GetKanbanData(req *opsdevmodel.OpsOperationEventKanbanSearchReq) (result g.Map, err error) {
  569. result = g.Map{}
  570. columns := []struct {
  571. Status string
  572. Name string
  573. Key string
  574. }{
  575. {opsdevmodel.EventStatusPending, "待处理", "10"},
  576. {opsdevmodel.EventStatusProcessing, "处理中(重点)", "20"},
  577. {opsdevmodel.EventStatusProcessingNormal, "处理中(普通)", "30"},
  578. {opsdevmodel.EventStatusTransfer, "转研发", "40"},
  579. {opsdevmodel.EventStatusSuspended, "挂起", "70"},
  580. }
  581. for _, col := range columns {
  582. db := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).Where(s.Dao.Columns.EventStatus, col.Status)
  583. if req.KeyWords != "" {
  584. switch req.SearchType {
  585. case "custName":
  586. db = db.Where(s.Dao.Columns.CustName+" like ?", "%"+req.KeyWords+"%")
  587. case "feedbackReporter":
  588. db = db.Where(s.Dao.Columns.FeedbackReporter+" like ?", "%"+req.KeyWords+"%")
  589. case "eventNo":
  590. db = db.Where(s.Dao.Columns.EventNo+" like ?", "%"+req.KeyWords+"%")
  591. default:
  592. db = db.Where(s.Dao.Columns.EventTitle+" like ?", "%"+req.KeyWords+"%")
  593. }
  594. }
  595. if req.EventType != "" {
  596. db = db.Where(s.Dao.Columns.EventType, req.EventType)
  597. }
  598. if req.PriorityLevel != "" {
  599. db = db.Where(s.Dao.Columns.PriorityLevel, req.PriorityLevel)
  600. }
  601. count, err := db.Count()
  602. if err != nil {
  603. g.Log().Error(err)
  604. return nil, myerrors.DbError("查询看板数据失败")
  605. }
  606. switch req.SortBy {
  607. case "feedbackDateDesc":
  608. db = db.Order(s.Dao.Columns.FeedbackDate + " desc")
  609. case "feedbackDateAsc":
  610. db = db.Order(s.Dao.Columns.FeedbackDate + " asc")
  611. case "custName":
  612. db = db.Order(s.Dao.Columns.CustName + " asc")
  613. default:
  614. db = db.Order(s.Dao.Columns.FeedbackDate + " desc")
  615. }
  616. var list []*opsdevmodel.OpsOperationEvent
  617. err = db.Scan(&list)
  618. if err != nil {
  619. g.Log().Error(err)
  620. return nil, myerrors.DbError("查询看板数据失败")
  621. }
  622. result[col.Key] = g.Map{
  623. "status": col.Key,
  624. "name": col.Name,
  625. "count": count,
  626. "list": list,
  627. }
  628. }
  629. return result, nil
  630. }
  631. // AssignOpsUser 接收/分配运维人员
  632. func (s *OperationService) AssignOpsUser(req *opsdevmodel.AssignOpsUserReq) error {
  633. event := new(opsdevmodel.OpsOperationEvent)
  634. err := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.Id).Scan(event)
  635. if err != nil {
  636. g.Log().Error(err)
  637. return myerrors.DbError("查询运维事件失败")
  638. }
  639. if event.Id <= 0 {
  640. return myerrors.TipsError("运维事件不存在")
  641. }
  642. if err := s.checkNotClosed(event); err != nil {
  643. return err
  644. }
  645. if event.EventStatus != opsdevmodel.EventStatusPending {
  646. return myerrors.TipsError("只有待处理状态的事件才能分配运维人员")
  647. }
  648. opsUserId := req.OpsUserId
  649. opsUserName := req.OpsUserName
  650. if opsUserId <= 0 {
  651. opsUserId = s.GetCxtUserId()
  652. }
  653. if opsUserName == "" {
  654. opsUserName = s.GetCxtUserName()
  655. }
  656. _, err = s.Dao.FieldsEx(service.UpdateFieldEx...).WherePri(s.Dao.Columns.Id, req.Id).Data(g.Map{
  657. s.Dao.Columns.EventStatus: opsdevmodel.EventStatusProcessing,
  658. s.Dao.Columns.OpsUserId: opsUserId,
  659. s.Dao.Columns.OpsUserName: opsUserName,
  660. s.Dao.Columns.AssignTime: gtime.Now(),
  661. s.Dao.Columns.UpdatedBy: s.GetCxtUserId(),
  662. s.Dao.Columns.UpdatedName: s.GetCxtUserName(),
  663. s.Dao.Columns.UpdatedTime: gtime.Now(),
  664. }).Update()
  665. if err != nil {
  666. g.Log().Error(err)
  667. return myerrors.DbError("分配运维人员失败")
  668. }
  669. return nil
  670. }
  671. // AddRecord 添加处理记录
  672. func (s *OperationService) AddRecord(req *opsdevmodel.AddRecordReq) error {
  673. if req.EventId <= 0 {
  674. return myerrors.ValidError("事件ID不能为空")
  675. }
  676. event := new(opsdevmodel.OpsOperationEvent)
  677. err := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).WherePri(s.Dao.Columns.Id, req.EventId).Scan(event)
  678. if err != nil {
  679. g.Log().Error(err)
  680. return myerrors.DbError("查询运维事件失败")
  681. }
  682. if event.Id <= 0 {
  683. return myerrors.TipsError("运维事件不存在")
  684. }
  685. if err := s.checkNotClosed(event); err != nil {
  686. return err
  687. }
  688. return s.Dao.Transaction(s.Ctx, func(ctx context.Context, tx *gdb.TX) error {
  689. recordData := g.Map{
  690. s.RecordDao.Columns.EventId: req.EventId,
  691. s.RecordDao.Columns.HandleUserId: s.GetCxtUserId(),
  692. s.RecordDao.Columns.HandleUserName: s.GetCxtUserName(),
  693. s.RecordDao.Columns.HandleContent: req.HandleContent,
  694. s.RecordDao.Columns.HandleResult: req.HandleResult,
  695. s.RecordDao.Columns.HandleDate: gtime.Now(),
  696. s.RecordDao.Columns.OperateType: opsdevmodel.OperateTypeProcess,
  697. }
  698. service.SetCreatedInfo(recordData, s.GetCxtUserId(), s.GetCxtUserName())
  699. result, err := tx.Model(s.RecordDao.Table).Data(recordData).Insert()
  700. if err != nil {
  701. g.Log().Error(err)
  702. return myerrors.DbError("添加处理记录失败")
  703. }
  704. recordId, err := result.LastInsertId()
  705. if err != nil {
  706. g.Log().Error(err)
  707. return myerrors.DbError("获取记录ID失败")
  708. }
  709. if len(req.Attachments) > 0 {
  710. for _, attachment := range req.Attachments {
  711. attachData := g.Map{
  712. s.AttachmentDao.Columns.EventId: req.EventId,
  713. s.AttachmentDao.Columns.EventRecordId: int(recordId),
  714. s.AttachmentDao.Columns.FileName: attachment.FileName,
  715. s.AttachmentDao.Columns.FileUrl: attachment.FileUrl,
  716. s.AttachmentDao.Columns.FileType: attachment.FileType,
  717. }
  718. service.SetCreatedInfo(attachData, s.GetCxtUserId(), s.GetCxtUserName())
  719. _, err := tx.Model(s.AttachmentDao.Table).Data(attachData).Insert()
  720. if err != nil {
  721. g.Log().Error(err)
  722. return myerrors.DbError("保存附件失败")
  723. }
  724. }
  725. }
  726. return nil
  727. })
  728. }
  729. // DeleteAttachment 删除附件
  730. func (s *OperationService) DeleteAttachment(req *comm_def.IdReq) error {
  731. if req.Id <= 0 {
  732. return myerrors.ValidError("参数有误!")
  733. }
  734. attachment := new(opsdevmodel.OpsOperationEventAttachment)
  735. err := s.AttachmentDao.WherePri(s.AttachmentDao.Columns.Id, req.Id).Scan(attachment)
  736. if err != nil {
  737. g.Log().Error(err)
  738. return myerrors.DbError("查询附件失败")
  739. }
  740. if attachment.Id <= 0 {
  741. return myerrors.TipsError("附件不存在")
  742. }
  743. _, err = s.AttachmentDao.WherePri(s.AttachmentDao.Columns.Id, req.Id).Data(g.Map{
  744. s.AttachmentDao.Columns.DeletedTime: gtime.Now(),
  745. s.AttachmentDao.Columns.UpdatedBy: s.GetCxtUserId(),
  746. s.AttachmentDao.Columns.UpdatedName: s.GetCxtUserName(),
  747. }).Update()
  748. if err != nil {
  749. g.Log().Error(err)
  750. return myerrors.DbError("删除附件失败")
  751. }
  752. return nil
  753. }
  754. // GetHistoryList 运维历史列表
  755. func (s *OperationService) GetHistoryList(req *opsdevmodel.OpsOperationEventHistorySearchReq) (total int, list []*opsdevmodel.OpsOperationEvent, err error) {
  756. db := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime)
  757. if !req.IncludeClosed {
  758. db = db.Where(s.Dao.Columns.EventStatus, opsdevmodel.EventStatusClosed)
  759. }
  760. if req.ScopeType == "my" {
  761. db = db.Where(s.Dao.Columns.OpsUserId, s.GetCxtUserId())
  762. }
  763. if req.EventTitle != "" {
  764. db = db.Where(s.Dao.Columns.EventTitle+" like ?", "%"+req.EventTitle+"%")
  765. }
  766. if req.CustName != "" {
  767. db = db.Where(s.Dao.Columns.CustName+" like ?", "%"+req.CustName+"%")
  768. }
  769. if req.FeedbackReporter != "" {
  770. db = db.Where(s.Dao.Columns.FeedbackReporter+" like ?", "%"+req.FeedbackReporter+"%")
  771. }
  772. if req.OpsUserName != "" {
  773. db = db.Where(s.Dao.Columns.OpsUserName+" like ?", "%"+req.OpsUserName+"%")
  774. }
  775. if req.BeginTime != "" {
  776. db = db.Where(s.Dao.Columns.FeedbackDate+" >= ?", req.BeginTime)
  777. }
  778. if req.EndTime != "" {
  779. db = db.Where(s.Dao.Columns.FeedbackDate+" <= ?", req.EndTime)
  780. }
  781. total, err = db.Count()
  782. if err != nil {
  783. g.Log().Error(err)
  784. return 0, nil, myerrors.DbError("查询运维历史数量失败")
  785. }
  786. pageNum, pageSize := req.GetPage()
  787. err = db.Page(pageNum, pageSize).Order(s.Dao.Columns.CompleteTime + " desc").Scan(&list)
  788. if err != nil {
  789. g.Log().Error(err)
  790. return 0, nil, myerrors.DbError("查询运维历史列表失败")
  791. }
  792. return total, list, nil
  793. }
  794. // Export 导出运维历史
  795. func (s *OperationService) Export(ctx context.Context, req *opsdevmodel.OpsOperationEventHistoryExport) (content *opsdevmodel.OpsOperationEventHistoryExportContent, err error) {
  796. db := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime)
  797. if !req.IncludeClosed {
  798. db = db.Where(s.Dao.Columns.EventStatus, opsdevmodel.EventStatusClosed)
  799. }
  800. if req.ScopeType == "my" {
  801. db = db.Where(s.Dao.Columns.OpsUserId, s.GetCxtUserId())
  802. }
  803. if req.EventTitle != "" {
  804. db = db.Where(s.Dao.Columns.EventTitle+" like ?", "%"+req.EventTitle+"%")
  805. }
  806. if req.CustName != "" {
  807. db = db.Where(s.Dao.Columns.CustName+" like ?", "%"+req.CustName+"%")
  808. }
  809. if req.FeedbackReporter != "" {
  810. db = db.Where(s.Dao.Columns.FeedbackReporter+" like ?", "%"+req.FeedbackReporter+"%")
  811. }
  812. if req.OpsUserName != "" {
  813. db = db.Where(s.Dao.Columns.OpsUserName+" like ?", "%"+req.OpsUserName+"%")
  814. }
  815. if req.BeginTime != "" {
  816. db = db.Where(s.Dao.Columns.FeedbackDate+" >= ?", req.BeginTime)
  817. }
  818. if req.EndTime != "" {
  819. db = db.Where(s.Dao.Columns.FeedbackDate+" <= ?", req.EndTime)
  820. }
  821. var list []*opsdevmodel.OpsOperationEvent
  822. err = db.Order(s.Dao.Columns.CompleteTime + " desc").Scan(&list)
  823. if err != nil {
  824. g.Log().Error(err)
  825. return nil, myerrors.DbError("查询运维历史失败")
  826. }
  827. // 获取所有事件ID
  828. var eventIds []int
  829. for _, item := range list {
  830. eventIds = append(eventIds, item.Id)
  831. }
  832. // 获取所有事件的处理记录
  833. var records []*opsdevmodel.OpsOperationEventRecord
  834. if len(eventIds) > 0 {
  835. err = s.RecordDao.WhereIn(s.RecordDao.Columns.EventId, eventIds).Order(s.RecordDao.Columns.HandleDate + " desc").Scan(&records)
  836. if err != nil {
  837. g.Log().Error(err)
  838. return nil, myerrors.DbError("查询处理记录失败")
  839. }
  840. }
  841. // 构建记录Map
  842. recordMap := make(map[int][]*opsdevmodel.OpsOperationEventRecord)
  843. for _, record := range records {
  844. recordMap[record.EventId] = append(recordMap[record.EventId], record)
  845. }
  846. // 构建状态Map
  847. statusMap := map[string]string{
  848. opsdevmodel.EventStatusPending: "待处理",
  849. opsdevmodel.EventStatusProcessing: "处理中(重点)",
  850. opsdevmodel.EventStatusProcessingNormal: "处理中(普通)",
  851. opsdevmodel.EventStatusTransfer: "转研发",
  852. opsdevmodel.EventStatusSuspended: "挂起",
  853. opsdevmodel.EventStatusClosed: "已关闭",
  854. }
  855. // 构建操作类型Map
  856. operateTypeMap := map[string]string{
  857. opsdevmodel.OperateTypeProcess: "处理",
  858. opsdevmodel.OperateTypeResume: "转处理",
  859. opsdevmodel.OperateTypeTransfer: "转研发",
  860. opsdevmodel.OperateTypeSuspend: "挂起",
  861. opsdevmodel.OperateTypeClose: "关闭",
  862. }
  863. // 构建处理结果Map
  864. handleResultMap := map[string]string{
  865. opsdevmodel.HandleResultResolved: "已解决",
  866. opsdevmodel.HandleResultPartially: "部分解决",
  867. opsdevmodel.HandleResultUnresolved: "未解决",
  868. }
  869. // 构建导出数据
  870. exportDataList := make([]map[string]interface{}, 0)
  871. for _, item := range list {
  872. // 构建处理过程
  873. processBuilder := ""
  874. if itemRecords, ok := recordMap[item.Id]; ok && len(itemRecords) > 0 {
  875. for i, record := range itemRecords {
  876. // 处理时间(仅年月)
  877. handleDate := ""
  878. if record.HandleDate != nil {
  879. handleDate = record.HandleDate.Format("Y-m-d H:i:s")
  880. }
  881. // 操作类型描述
  882. operateTypeStr := operateTypeMap[record.OperateType]
  883. if operateTypeStr == "" {
  884. operateTypeStr = record.OperateType
  885. }
  886. // 处理结果描述
  887. handleResultStr := handleResultMap[record.HandleResult]
  888. if handleResultStr == "" {
  889. handleResultStr = record.HandleResult
  890. }
  891. // 清理处理内容中的HTML标签
  892. cleanContent := cleanHtmlTags(record.HandleContent)
  893. recordStr := fmt.Sprintf("%s-%s-%s-%s-%s", handleDate, record.HandleUserName, operateTypeStr, handleResultStr, cleanContent)
  894. if i > 0 {
  895. processBuilder += "\n" + recordStr
  896. } else {
  897. processBuilder = recordStr
  898. }
  899. }
  900. }
  901. closedTime := ""
  902. if item.CompleteTime != nil {
  903. closedTime = item.CompleteTime.Format("Y-m-d")
  904. }
  905. feedbackDate := ""
  906. if item.FeedbackDate != nil {
  907. feedbackDate = item.FeedbackDate.Format("Y-m-d")
  908. }
  909. exportData := map[string]interface{}{
  910. "eventNo": item.EventNo,
  911. "eventTitle": item.EventTitle,
  912. "eventStatus": statusMap[item.EventStatus],
  913. "eventType": item.EventType,
  914. "custName": item.CustName,
  915. "opsUserName": item.OpsUserName,
  916. "feedbackDate": feedbackDate,
  917. "priorityLevel": item.PriorityLevel,
  918. "handleProcess": processBuilder,
  919. "closedTime": closedTime,
  920. }
  921. exportDataList = append(exportDataList, exportData)
  922. }
  923. // 调用公共导出方法
  924. exportContent := new(opsdevmodel.OpsOperationEventHistoryExportContent)
  925. contentBase64, err := service.CommonExportExcel(ctx, "运维历史", opsdevmodel.OpsOperationEventHistoryExportData{}, exportDataList)
  926. if err != nil {
  927. g.Log().Error(err)
  928. return nil, myerrors.DbError("导出运维历史失败")
  929. }
  930. exportContent.Content = contentBase64
  931. return exportContent, nil
  932. }
  933. func (s *OperationService) ExportNonClosed(ctx context.Context, req *opsdevmodel.OpsOperationEventExport) (content *opsdevmodel.OpsOperationEventHistoryExportContent, err error) {
  934. db := s.Dao.FieldsEx(s.Dao.Columns.DeletedTime).Where(s.Dao.Columns.EventStatus+" != ?", opsdevmodel.EventStatusClosed)
  935. var list []*opsdevmodel.OpsOperationEvent
  936. err = db.Order(s.Dao.Columns.FeedbackDate + " desc").Scan(&list)
  937. if err != nil {
  938. g.Log().Error(err)
  939. return nil, myerrors.DbError("查询运维事件失败")
  940. }
  941. var eventIds []int
  942. for _, item := range list {
  943. eventIds = append(eventIds, item.Id)
  944. }
  945. var records []*opsdevmodel.OpsOperationEventRecord
  946. if len(eventIds) > 0 {
  947. err = s.RecordDao.WhereIn(s.RecordDao.Columns.EventId, eventIds).Order(s.RecordDao.Columns.HandleDate + " desc").Scan(&records)
  948. if err != nil {
  949. g.Log().Error(err)
  950. return nil, myerrors.DbError("查询处理记录失败")
  951. }
  952. }
  953. recordMap := make(map[int][]*opsdevmodel.OpsOperationEventRecord)
  954. for _, record := range records {
  955. recordMap[record.EventId] = append(recordMap[record.EventId], record)
  956. }
  957. statusMap := map[string]string{
  958. opsdevmodel.EventStatusPending: "待处理",
  959. opsdevmodel.EventStatusProcessing: "处理中(重点)",
  960. opsdevmodel.EventStatusProcessingNormal: "处理中(普通)",
  961. opsdevmodel.EventStatusTransfer: "转研发",
  962. opsdevmodel.EventStatusSuspended: "挂起",
  963. opsdevmodel.EventStatusClosed: "已关闭",
  964. }
  965. operateTypeMap := map[string]string{
  966. opsdevmodel.OperateTypeProcess: "处理",
  967. opsdevmodel.OperateTypeResume: "转处理",
  968. opsdevmodel.OperateTypeTransfer: "转研发",
  969. opsdevmodel.OperateTypeSuspend: "挂起",
  970. opsdevmodel.OperateTypeClose: "关闭",
  971. }
  972. handleResultMap := map[string]string{
  973. opsdevmodel.HandleResultResolved: "已解决",
  974. opsdevmodel.HandleResultPartially: "部分解决",
  975. opsdevmodel.HandleResultUnresolved: "未解决",
  976. }
  977. exportDataList := make([]map[string]interface{}, 0)
  978. for _, item := range list {
  979. processBuilder := ""
  980. if itemRecords, ok := recordMap[item.Id]; ok && len(itemRecords) > 0 {
  981. for i, record := range itemRecords {
  982. handleDate := ""
  983. if record.HandleDate != nil {
  984. handleDate = record.HandleDate.Format("Y-m-d H:i:s")
  985. }
  986. operateTypeStr := operateTypeMap[record.OperateType]
  987. if operateTypeStr == "" {
  988. operateTypeStr = record.OperateType
  989. }
  990. handleResultStr := handleResultMap[record.HandleResult]
  991. if handleResultStr == "" {
  992. handleResultStr = record.HandleResult
  993. }
  994. cleanContent := cleanHtmlTags(record.HandleContent)
  995. recordStr := fmt.Sprintf("%s-%s-%s-%s-%s", handleDate, record.HandleUserName, operateTypeStr, handleResultStr, cleanContent)
  996. if i > 0 {
  997. processBuilder += "\n" + recordStr
  998. } else {
  999. processBuilder = recordStr
  1000. }
  1001. }
  1002. }
  1003. feedbackDate := ""
  1004. if item.FeedbackDate != nil {
  1005. feedbackDate = item.FeedbackDate.Format("Y-m-d")
  1006. }
  1007. exportData := map[string]interface{}{
  1008. "eventNo": item.EventNo,
  1009. "eventTitle": item.EventTitle,
  1010. "eventStatus": statusMap[item.EventStatus],
  1011. "eventType": item.EventType,
  1012. "custName": item.CustName,
  1013. "opsUserName": item.OpsUserName,
  1014. "feedbackDate": feedbackDate,
  1015. "priorityLevel": item.PriorityLevel,
  1016. "handleProcess": processBuilder,
  1017. }
  1018. exportDataList = append(exportDataList, exportData)
  1019. }
  1020. exportContent := new(opsdevmodel.OpsOperationEventHistoryExportContent)
  1021. contentBase64, err := service.CommonExportExcel(ctx, "运维事件", opsdevmodel.OpsOperationEventExportData{}, exportDataList)
  1022. if err != nil {
  1023. g.Log().Error(err)
  1024. return nil, myerrors.DbError("导出运维事件失败")
  1025. }
  1026. exportContent.Content = contentBase64
  1027. return exportContent, nil
  1028. }
  1029. func cleanHtmlTags(content string) string {
  1030. if content == "" {
  1031. return ""
  1032. }
  1033. imgPattern := regexp.MustCompile(`src=["']([^"']+)["']`)
  1034. imgMatches := imgPattern.FindAllStringSubmatch(content, -1)
  1035. seen := make(map[string]bool)
  1036. var imgUrls []string
  1037. for _, match := range imgMatches {
  1038. if len(match) > 1 && !seen[match[1]] {
  1039. seen[match[1]] = true
  1040. imgUrls = append(imgUrls, match[1])
  1041. }
  1042. }
  1043. htmlTagPattern := regexp.MustCompile(`<[^>]*>`)
  1044. result := htmlTagPattern.ReplaceAllString(content, "")
  1045. result = strings.ReplaceAll(result, "&nbsp;", " ")
  1046. result = strings.ReplaceAll(result, "&amp;", "&")
  1047. result = strings.ReplaceAll(result, "&lt;", "<")
  1048. result = strings.ReplaceAll(result, "&gt;", ">")
  1049. result = strings.ReplaceAll(result, "&quot;", "\"")
  1050. result = strings.ReplaceAll(result, "&#39;", "'")
  1051. result = strings.TrimSpace(result)
  1052. if len(imgUrls) > 0 {
  1053. result += "\n图片: "
  1054. for i, url := range imgUrls {
  1055. if i > 0 {
  1056. result += ", "
  1057. }
  1058. result += url
  1059. }
  1060. }
  1061. return result
  1062. }