task.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. package bo
  2. import (
  3. "errors"
  4. "fmt"
  5. "scd_check_tools/logger"
  6. "scd_check_tools/models/enum"
  7. "scd_check_tools/tools"
  8. "strconv"
  9. "strings"
  10. "sync"
  11. "github.com/astaxie/beego/orm"
  12. )
  13. //检测任务模型
  14. type T_data_task struct {
  15. Id int `orm:"pk"`
  16. Name string // '任务名称' ,
  17. Code string // '任务编号' ,
  18. StationId int // '所属变电站' ,
  19. ScdId int64 // 'SCD文件编号' ,
  20. ReportId int // '报告模板ID' ,
  21. Memo string // '任务说明' ,
  22. State int // '检测状态;0 未检查 1 检测中 2检测结束 3 异常中断' ,
  23. StartTime string // '开始检测时间' ,
  24. EndTime string // '检测结束时间' ,
  25. Cr int // '创建人' ,
  26. Ct string `orm:"-"` // '创建时间' ,
  27. Ur int // '更新人' ,
  28. Ut string `orm:"-"` // '更新时间'
  29. }
  30. //检测任务管理对象
  31. type TaskMgr struct {
  32. Model T_data_task
  33. DeviceBaseModel
  34. }
  35. var TaskSCD = sync.Map{}
  36. var modelDesc = "检测任务"
  37. //已终止的任务
  38. var stopedTaskMap = sync.Map{}
  39. func init() {
  40. orm.RegisterModel(new(T_data_task))
  41. }
  42. //重置检测异常中断的任务状态。一般在系统启动时处理
  43. func (c *TaskMgr) ChangeErr2Normal() {
  44. db := orm.NewOrm()
  45. db.Raw("update t_data_task set state=0 where state=1").Exec()
  46. }
  47. //设置指定任务的状态 flag:0 未检查 1 检测中 2检测结束 3 异常中断
  48. func (c *TaskMgr) SetActive(flag int) (err error) {
  49. dblog := new(SystemLog)
  50. dblog.SetUserInfo(c.GetUserInfo())
  51. dblog.Audittype = enum.AuditType_check_task
  52. dblog.Logtype = enum.LogType_Execute
  53. dblog.Eventtype = enum.OptEventType_Bus
  54. dblog.Eventlevel = enum.OptEventLevel_Hight
  55. db := orm.NewOrm()
  56. err = db.Read(&c.Model)
  57. if err != nil {
  58. logger.Logger.Error(err)
  59. return err
  60. }
  61. if flag == 0 {
  62. c.Model.StartTime = "1970-01-01 00:00:00"
  63. c.Model.EndTime = "1970-01-01 00:00:00"
  64. }
  65. if flag == 1 {
  66. if c.Model.State == 1 {
  67. return errors.New("当前任务正在检测中")
  68. }
  69. c.Model.StartTime = tools.NowTime()
  70. c.Model.EndTime = "1970-01-01 00:00:00"
  71. }
  72. if flag == 2 {
  73. c.Model.EndTime = tools.NowTime()
  74. }
  75. c.Model.State = flag
  76. _, err = db.Update(&c.Model)
  77. if err != nil {
  78. logger.Logger.Error(err)
  79. dblog.Description = fmt.Sprintf("更新%s%s状态失败:%s,操作数据:%+v", modelDesc, c.Model.Name, err.Error(), c.Model)
  80. dblog.Fail2()
  81. } else {
  82. dblog.Description = fmt.Sprintf("更新%s%s状态,操作数据:%+v", modelDesc, c.Model.Name, c.Model)
  83. dblog.Success2()
  84. if flag == 0 {
  85. return nil
  86. }
  87. if flag == 1 {
  88. return c.start(c.Model)
  89. }
  90. if flag == 2 {
  91. return c.finish(c.Model)
  92. }
  93. if flag == 3 {
  94. return c.stop(c.Model)
  95. }
  96. }
  97. return err
  98. }
  99. //开始检测
  100. func (c *TaskMgr) start(task T_data_task) error {
  101. scdParse := new(ScdParse)
  102. scdParse.SetUserInfo(c.GetUserInfo())
  103. scdMgr := new(ScdMgr)
  104. scdMgr.SetUserInfo(c.GetUserInfo())
  105. scdInfo, err := scdMgr.One(fmt.Sprintf("%d", task.ScdId))
  106. if err != nil {
  107. return err
  108. }
  109. c.Model.Id = task.Id
  110. _, cnt, _ := c.GetModels()
  111. if cnt == 0 {
  112. return errors.New("该检测任务还未配置检测模型")
  113. }
  114. TaskSCD.Store(task.ScdId, task)
  115. db := orm.NewOrm()
  116. //清除原步骤及状态
  117. db.Raw("delete from t_data_task_check_step where task_id=?", task.Id).Exec()
  118. db.Raw("delete from t_scd_node_rule_parse where scd_id=?", task.ScdId).Exec()
  119. db.Raw("delete from t_scd_fcda_check_result where scd_id=?", task.ScdId).Exec()
  120. rept := new(TaskReportMgr)
  121. rept.Model = T_data_task_report{TaskId: task.Id}
  122. rept.Delete()
  123. //生成新的步骤
  124. db.Raw("insert into t_data_task_check_step(task_id,task_step_id,state) select ?,id,0 from global_const_code where parentcode='CHECK_STEP' order by id", task.Id).Exec()
  125. var isStop = func(taskid int) bool {
  126. v, r := stopedTaskMap.Load(taskid)
  127. if r {
  128. return v.(bool)
  129. }
  130. return false
  131. }
  132. go func(task T_data_task, scdInfo orm.Params) {
  133. _, err = scdParse.XmlIEDParse(tools.IsEmpty(scdInfo["station_id"]), tools.IsEmpty(scdInfo["path"]), tools.IsEmpty(scdInfo["scd_name"]))
  134. if isStop(task.Id) {
  135. return
  136. }
  137. if err == nil {
  138. scdParse.XmlCheckParse(task.ScdId)
  139. if isStop(task.Id) {
  140. return
  141. }
  142. //模型分析
  143. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_model_parse.Code(), 1)
  144. checkAreaMgr := new(CheckAreaMgr)
  145. checkAreaMgr.ScdId = task.ScdId
  146. err = checkAreaMgr.CheckAreaIedRelation()
  147. if isStop(task.Id) {
  148. return
  149. }
  150. if err != nil {
  151. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_model_parse.Code(), 3, err.Error())
  152. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_fcda_parse.Code(), 3, "终止")
  153. } else {
  154. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_model_parse.Code(), 2)
  155. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_fcda_parse.Code(), 1)
  156. err = checkAreaMgr.CheckIedFcda()
  157. if isStop(task.Id) {
  158. return
  159. }
  160. if err != nil {
  161. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_fcda_parse.Code(), 3, err.Error())
  162. } else {
  163. c.SetStep(tools.IsEmpty(task.ScdId), enum.TaskStep_SCD_fcda_parse.Code(), 2)
  164. }
  165. }
  166. }
  167. }(task, scdInfo)
  168. return err
  169. }
  170. //取消/终止检测
  171. func (c *TaskMgr) stop(task T_data_task) error {
  172. TaskSCD.Delete(task.ScdId)
  173. stopedTaskMap.Store(task.Id, true)
  174. db := orm.NewOrm()
  175. _, err := db.Raw("update t_data_task_check_step set state=3,ut=now() where task_id=? and state<2", task.Id).Exec()
  176. return err
  177. }
  178. //完成检测
  179. func (c *TaskMgr) finish(task T_data_task) error {
  180. db := orm.NewOrm()
  181. _, err := db.Raw("update t_data_task_check_step set state=2,ut=now() where task_id=? and state=0", task.Id).Exec()
  182. return err
  183. }
  184. //更新检测步骤状态
  185. func (c *TaskMgr) SetStep(scdid string, stepcode string, state int, msg ...string) {
  186. db := orm.NewOrm()
  187. scdidint64, _ := strconv.ParseInt(scdid, 10, 64)
  188. taskid, has := TaskSCD.Load(scdidint64)
  189. if !has {
  190. return
  191. }
  192. errormsg := ""
  193. if len(msg) > 0 {
  194. errormsg = msg[0]
  195. }
  196. t1 := "ct" //步骤开始时间字段
  197. if state > 1 {
  198. t1 = "ut" //步骤结束时间字段
  199. }
  200. db.Raw("update t_data_task_check_step set state=?,state_msg=?,"+t1+"=now() where task_id=? and task_step_id=(select id from global_const_code where parentcode='CHECK_STEP' and code=?)", state, errormsg, taskid, stepcode).Exec()
  201. // 所有步骤都不为0和1了,则分析完成
  202. hasStep := []orm.Params{}
  203. _, err := db.Raw("select count(1) cnt from t_data_task_check_step where task_id=? and state in(0,1)", taskid).Values(&hasStep)
  204. if err != nil {
  205. logger.Logger.Error(err)
  206. return
  207. }
  208. if len(hasStep) == 0 || tools.IsEmpty(hasStep[0]["cnt"]) == "0" {
  209. // 全部步骤检测完成,更改任务检测状态为已完成
  210. db.Raw("update t_data_task set state=2 where id=?", taskid).Exec()
  211. }
  212. }
  213. //保存任务信息
  214. func (c *TaskMgr) Save() (id int, err error) {
  215. dblog := new(SystemLog)
  216. dblog.SetUserInfo(c.GetUserInfo())
  217. dblog.Audittype = enum.AuditType_check_task
  218. dblog.Logtype = enum.LogType_Insert
  219. dblog.Eventtype = enum.OptEventType_Bus
  220. dblog.Eventlevel = enum.OptEventLevel_Hight
  221. db := orm.NewOrm()
  222. oldinfo := T_data_task{}
  223. oldid, _ := c.ExistName(c.Model.Name)
  224. if c.Model.Id == 0 {
  225. if oldid != "" {
  226. return 0, errors.New("任务名称(" + c.Model.Name + ")已存在")
  227. }
  228. c.Model.StartTime = "1970-01-01 00:00:00"
  229. c.Model.EndTime = "1970-01-01 00:00:00"
  230. c.Model.State = 0
  231. c.Model.Cr, _ = strconv.Atoi(c.GetUserId())
  232. id64 := int64(0)
  233. id64, err = db.Insert(&c.Model)
  234. id = int(id64)
  235. } else {
  236. if oldid != "" && oldid != tools.IsEmpty(c.Model.Id) {
  237. return 0, errors.New("任务名称(" + c.Model.Name + ")已存在")
  238. }
  239. id = c.Model.Id
  240. oldinfo, _ = c.One()
  241. if oldinfo.State == 1 {
  242. return 0, errors.New("任务正在检测中,不允许编辑")
  243. }
  244. c.Model.StartTime = oldinfo.StartTime
  245. c.Model.EndTime = oldinfo.EndTime
  246. c.Model.State = oldinfo.State
  247. c.Model.Ur, _ = strconv.Atoi(c.GetUserId())
  248. _, err = db.Update(&c.Model)
  249. }
  250. if err != nil {
  251. logger.Logger.Error(err)
  252. dblog.Description = fmt.Sprintf("保存%s信息失败:%s,操作数据:%+v", modelDesc, err.Error(), c.Model)
  253. dblog.Fail2()
  254. } else {
  255. dblog.Description = fmt.Sprintf("保存%s信息成功,操作数据:%+v", modelDesc, c.Model)
  256. dblog.Success2()
  257. if oldinfo.Id == 0 || oldinfo.ScdId != c.Model.ScdId {
  258. // 解析模型间隔及标准装置
  259. checkArea := new(CheckAreaMgr)
  260. checkArea.ScdId = c.Model.ScdId
  261. checkArea.Reset()
  262. }
  263. }
  264. return id, err
  265. }
  266. func (c *TaskMgr) One() (T_data_task, error) {
  267. if c.Model.Id == 0 {
  268. return c.Model, errors.New("未指定id")
  269. }
  270. o := orm.NewOrm()
  271. tmp := T_data_task{Id: c.Model.Id}
  272. err := o.Read(&tmp)
  273. if err != nil {
  274. logger.Logger.Error(err)
  275. }
  276. return tmp, err
  277. }
  278. func (c *TaskMgr) ExistName(name string) (string, error) {
  279. o := orm.NewOrm()
  280. rowset := []orm.Params{}
  281. _, err := o.Raw("select id from t_data_task where name=?", name).Values(&rowset)
  282. if err != nil {
  283. logger.Logger.Error(err)
  284. return "", err
  285. }
  286. if len(rowset) > 0 {
  287. return tools.IsEmpty(rowset[0]["id"]), nil
  288. }
  289. return "", nil
  290. }
  291. //根据model中指定的id删除检测任务
  292. func (c *TaskMgr) Delete() (err error) {
  293. dblog := new(SystemLog)
  294. u := c.GetUserInfo()
  295. dblog.SetUserInfo(u)
  296. dblog.Audittype = enum.AuditType_check_task
  297. dblog.Logtype = enum.LogType_Delete
  298. dblog.Eventtype = enum.OptEventType_Bus
  299. dblog.Eventlevel = enum.OptEventLevel_Hight
  300. db := orm.NewOrm()
  301. if c.Model.Id == 0 {
  302. return errors.New("无效的ID")
  303. } else {
  304. db.Read(&c.Model)
  305. if c.Model.State == 1 {
  306. return errors.New("任务正在检测中,不允许删除")
  307. }
  308. _, err = db.Delete(&c.Model)
  309. }
  310. if err != nil {
  311. logger.Logger.Error(err)
  312. dblog.Description = fmt.Sprintf("删除%s%s失败:%s", modelDesc, c.Model.Name, err.Error())
  313. dblog.Fail2()
  314. } else {
  315. TaskSCD.Delete(c.Model.ScdId)
  316. dblog.Description = fmt.Sprintf("删除%s%s成功", modelDesc, c.Model.Name)
  317. dblog.Success2()
  318. //同步删除关联数据
  319. db.Raw("delete from t_data_task_check_step where task_id=?", c.Model.Id).Exec()
  320. db.Raw("delete from t_data_check_area where scd_id=?", c.Model.ScdId).Exec()
  321. db.Raw("delete from t_data_check_area_ied where scd_id=?", c.Model.ScdId).Exec()
  322. db.Raw("delete from t_scd_node_rule_parse where scd_id=?", c.Model.ScdId).Exec()
  323. db.Raw("delete from t_scd_fcda_check_result where scd_id=?", c.Model.ScdId).Exec()
  324. scdmgr := new(ScdMgr)
  325. scdmgr.SetUserInfo(u)
  326. scdmgr.DeleteScd(strconv.FormatInt(c.Model.ScdId, 10), true)
  327. //删除关联模型
  328. m1 := new(TaskModelMgr)
  329. m1.SetUserInfo(u)
  330. m1.Model = T_data_task_model{TaskId: c.Model.Id}
  331. m1.Delete()
  332. //删除报告关联数据
  333. r1 := new(TaskReportMgr)
  334. r1.SetUserInfo(u)
  335. r1.Model = T_data_task_report{TaskId: c.Model.Id}
  336. r1.Delete()
  337. }
  338. return err
  339. }
  340. //根据model中的指定过滤属性条件查询任务列表
  341. func (c *TaskMgr) List(pageno, pagesize int) ([]orm.Params, int, error) {
  342. dblog := new(SystemLog)
  343. dblog.SetUserInfo(c.GetUserInfo())
  344. dblog.Audittype = enum.AuditType_check_task
  345. dblog.Logtype = enum.LogType_Query
  346. dblog.Eventtype = enum.OptEventType_Bus
  347. dblog.Eventlevel = enum.OptEventLevel_Low
  348. o := orm.NewOrm()
  349. sqlParamters := []interface{}{}
  350. sql := "select t.*,u.name username,s.scd_name,ts.area_name station_name,c1.id voltage_level,c1.name voltage_name,ifnull(r1.doc,'') report_doc from t_data_task t left join t_scd_scl s on t.scd_id=s.id left join t_data_user u on t.cr=u.id left join t_data_area ts on t.station_id=ts.ID left join global_const_code c1 on ts.AREA_LEVEL=c1.id and c1.parentcode='voltage_level' left join t_data_task_report r1 on t.id=r1.task_id where 1=1 "
  351. if c.Model.Id > 0 {
  352. sql += " and t.id=?"
  353. sqlParamters = append(sqlParamters, c.Model.Id)
  354. }
  355. if c.Model.StationId > 0 {
  356. sql += " and t.station_id=?"
  357. sqlParamters = append(sqlParamters, c.Model.StationId)
  358. }
  359. if c.Model.Name != "" {
  360. sql += " and t.name like ?"
  361. sqlParamters = append(sqlParamters, "%"+c.Model.Name+"%")
  362. }
  363. if c.Model.ScdId > 0 {
  364. sql += " and t.scd_id=?"
  365. sqlParamters = append(sqlParamters, c.Model.ScdId)
  366. }
  367. if c.Model.State >= 0 {
  368. sql += " and t.state=?"
  369. sqlParamters = append(sqlParamters, c.Model.State)
  370. }
  371. if c.Model.StartTime != "" {
  372. sql += " and t.ct>=?"
  373. sqlParamters = append(sqlParamters, c.Model.StartTime+" 00:00:00")
  374. }
  375. if c.Model.EndTime != "" {
  376. sql += " and t.ct<=?"
  377. sqlParamters = append(sqlParamters, c.Model.EndTime+" 23:59:59")
  378. }
  379. limit := fmt.Sprintf(" order by t.ct desc limit %d,%d", (pageno-1)*pagesize, pagesize)
  380. r := []orm.Params{}
  381. _, err := o.Raw(sql+limit, sqlParamters).Values(&r)
  382. dblog.Description = fmt.Sprintf("SQL:%s 参数:%+v", sql+limit, sqlParamters)
  383. if err != nil {
  384. logger.Logger.Error(err, dblog.Description)
  385. dblog.Fail2()
  386. return nil, 0, err
  387. }
  388. dblog.Success2()
  389. total := []orm.Params{}
  390. _, err = o.Raw(strings.Replace(sql, "t.*,u.name username,s.scd_name,ts.area_name station_name", "count(1) cnt", 1), sqlParamters).Values(&total)
  391. if err != nil {
  392. logger.Logger.Error(err)
  393. return nil, 0, err
  394. }
  395. totalCnt := 0
  396. if len(total) > 0 {
  397. totalCnt, _ = strconv.Atoi(tools.IsEmpty(total[0]["cnt"]))
  398. }
  399. return r, totalCnt, err
  400. }
  401. //获取指定任务的已选择模型对应的系统内置模型
  402. func (c *TaskMgr) GetModels() ([]orm.Params, int, error) {
  403. if c.Model.Id == 0 {
  404. return nil, 0, errors.New("任务ID不能为空")
  405. }
  406. db := orm.NewOrm()
  407. rowset := []orm.Params{}
  408. _, err := db.Raw("select t.*,t1.from_model_id sys_model_id,t2.model_name,t2.vol_id,t2.line_link_style from t_data_task_model t,t_data_model_defualt t1,t_data_model_defualt t2 where t.model_id=t1.id and t1.from_model_id=t2.id and t.task_id=?", c.Model.Id).Values(&rowset)
  409. if err != nil {
  410. return rowset, 0, err
  411. }
  412. return rowset, len(rowset), nil
  413. }
  414. //根据SCD ID获取指定任务的已选择模型
  415. func (c *TaskMgr) GetModelsByScdID(scdid int64) ([]orm.Params, int, error) {
  416. if scdid == 0 {
  417. return nil, 0, errors.New("SCD ID不能为空")
  418. }
  419. db := orm.NewOrm()
  420. rowset := []orm.Params{}
  421. _, err := db.Raw("select t.*,t1.model_name,t1.vol_id,t1.line_link_style from t_data_task a, t_data_task_model t,t_data_model_defualt t1 where a.id=t.task_id and t.model_id=t1.id and a.scd_id=?", scdid).Values(&rowset)
  422. if err != nil {
  423. return rowset, 0, err
  424. }
  425. return rowset, len(rowset), nil
  426. }
  427. func (c *TaskMgr) GetCheckStepInfo() ([]orm.Params, int, error) {
  428. db := orm.NewOrm()
  429. rowset := []orm.Params{}
  430. _, err := db.Raw("select a.*,a1.name step_name,case a.state when 0 then '未开始' when 1 then '检测中' when 2 then '已完成' when 3 then '检测终止' else '检测异常' end state_name from t_data_task_check_step a left join global_const_code a1 on a.task_step_id=a1.id and a1.parentcode='CHECK_STEP' where a.task_id=? order by a1.id ", c.Model.Id).Values(&rowset)
  431. if err != nil {
  432. return rowset, 0, err
  433. }
  434. return rowset, len(rowset), nil
  435. }