GAAS GFrame项目web后台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

299 lines
9.0 KiB

  1. package models
  2. import (
  3. "LAPP_GAAS_GFrame_BACKEND/db"
  4. "errors"
  5. "strings"
  6. "time"
  7. "github.com/go-xorm/xorm"
  8. )
  9. type TaskProtocol int8
  10. const (
  11. TaskHTTP TaskProtocol = iota + 1 // HTTP协议
  12. TaskRPC // RPC方式执行命令
  13. )
  14. type TaskLevel int8
  15. const (
  16. TaskLevelParent TaskLevel = 1 // 父任务
  17. TaskLevelChild TaskLevel = 2 // 子任务(依赖任务)
  18. )
  19. type TaskDependencyStatus int8
  20. const (
  21. TaskDependencyStatusStrong TaskDependencyStatus = 1 // 强依赖
  22. TaskDependencyStatusWeak TaskDependencyStatus = 2 // 弱依赖
  23. )
  24. type TaskHTTPMethod int8
  25. const (
  26. TaskHTTPMethodGet TaskHTTPMethod = 1
  27. TaskHttpMethodPost TaskHTTPMethod = 2
  28. )
  29. type Status int8
  30. type CommonMap map[string]interface{}
  31. const (
  32. Disabled Status = 0 // 禁用
  33. Failure Status = 0 // 失败
  34. Enabled Status = 1 // 启用
  35. Running Status = 1 // 运行中
  36. Finish Status = 2 // 完成
  37. Cancel Status = 3 // 取消
  38. )
  39. const DefaultTimeFormat = "2006-01-02 15:04:05"
  40. // 任务
  41. type Task struct {
  42. Id int `json:"id" xorm:"int pk autoincr"`
  43. Name string `json:"name" xorm:"varchar(32) notnull"` // 任务名称
  44. Level TaskLevel `json:"level" xorm:"tinyint notnull index default 1"` // 任务等级 1: 主任务 2: 依赖任务
  45. DependencyTaskId string `json:"dependency_task_id" xorm:"varchar(64) notnull default ''"` // 依赖任务ID,多个ID逗号分隔
  46. DependencyStatus TaskDependencyStatus `json:"dependency_status" xorm:"tinyint notnull default 1"` // 依赖关系 1:强依赖 主任务执行成功, 依赖任务才会被执行 2:弱依赖
  47. Spec string `json:"spec" xorm:"varchar(64) notnull"` // crontab
  48. Protocol TaskProtocol `json:"protocol" xorm:"tinyint notnull index"` // 协议 1:http 2:系统命令
  49. Command string `json:"command" xorm:"varchar(256) notnull"` // URL地址或shell命令
  50. HttpMethod TaskHTTPMethod `json:"http_method" xorm:"tinyint notnull default 1"` // http请求方法
  51. Timeout int `json:"timeout" xorm:"mediumint notnull default 0"` // 任务执行超时时间(单位秒),0不限制
  52. Multi int8 `json:"multi" xorm:"tinyint notnull default 1"` // 是否允许多实例运行
  53. RetryTimes int8 `json:"retry_times" xorm:"tinyint notnull default 0"` // 重试次数
  54. RetryInterval int16 `json:"retry_interval" xorm:"smallint notnull default 0"` // 重试间隔时间
  55. NotifyStatus int8 `json:"notify_status" xorm:"tinyint notnull default 1"` // 任务执行结束是否通知 0: 不通知 1: 失败通知 2: 执行结束通知 3: 任务执行结果关键字匹配通知
  56. NotifyType int8 `json:"notify_type" xorm:"tinyint notnull default 0"` // 通知类型 1: 邮件 2: slack 3: webhook
  57. NotifyReceiverId string `json:"notify_receiver_id" xorm:"varchar(256) notnull default '' "` // 通知接受者ID, setting表主键ID,多个ID逗号分隔
  58. NotifyKeyword string `json:"notify_keyword" xorm:"varchar(128) notnull default '' "`
  59. Tag string `json:"tag" xorm:"varchar(32) notnull default ''"`
  60. Remark string `json:"remark" xorm:"varchar(100) notnull default ''"` // 备注
  61. Status Status `json:"status" xorm:"tinyint notnull index default 0"` // 状态 1:正常 0:停止
  62. Created time.Time `json:"created" xorm:"datetime notnull created"` // 创建时间
  63. Deleted time.Time `json:"deleted" xorm:"datetime deleted"` // 删除时间
  64. Hosts []TaskHostDetail `json:"hosts" xorm:"-"`
  65. NextRunTime time.Time `json:"next_run_time" xorm:"-"`
  66. }
  67. func taskHostTableName() []string {
  68. return []string{"task_host", "th"}
  69. }
  70. // 新增
  71. func (t *Task) Create() (insertId int, err error) {
  72. e := db.Eloquent.Master()
  73. _, err = e.Table("task").Insert(t)
  74. if err == nil {
  75. insertId = t.Id
  76. }
  77. return
  78. }
  79. func (t *Task) UpdateBean(id int) (int64, error) {
  80. e := db.Eloquent.Master()
  81. return e.ID(id).
  82. Cols(`name,spec,protocol,command,timeout,multi,
  83. retry_times,retry_interval,remark,notify_status,
  84. notify_type,notify_receiver_id, dependency_task_id, dependency_status, tag,http_method, notify_keyword`).
  85. Update(t)
  86. }
  87. // 更新
  88. func (t *Task) Update(id int, data CommonMap) (int64, error) {
  89. e := db.Eloquent.Master()
  90. return e.Table(t).ID(id).Update(data)
  91. }
  92. // 删除
  93. func (t *Task) Delete(id int) (int64, error) {
  94. e := db.Eloquent.Master()
  95. return e.Id(id).Delete(t)
  96. }
  97. // 禁用
  98. func (t *Task) Disable(id int) (int64, error) {
  99. return t.Update(id, CommonMap{"status": Disabled})
  100. }
  101. // 激活
  102. func (t *Task) Enable(id int) (int64, error) {
  103. return t.Update(id, CommonMap{"status": Enabled})
  104. }
  105. // 获取所有激活任务
  106. func (t *Task) ActiveList(pageSize int, pageIndex int) ([]Task, error) {
  107. e := db.Eloquent.Master()
  108. Offset := (pageIndex - 1) * pageSize
  109. list := make([]Task, 0)
  110. err := e.Where("status = ? AND level = ?", Enabled, TaskLevelParent).Limit(pageSize, Offset).
  111. Find(&list)
  112. if err != nil {
  113. return list, err
  114. }
  115. return t.setHostsForTasks(list)
  116. }
  117. // 获取某个主机下的所有激活任务
  118. func (t *Task) ActiveListByHostId(hostId int16) ([]Task, error) {
  119. e := db.Eloquent.Master()
  120. taskHostModel := new(TaskHost)
  121. taskIds, err := taskHostModel.GetTaskIdsByHostId(hostId)
  122. if err != nil {
  123. return nil, err
  124. }
  125. if len(taskIds) == 0 {
  126. return nil, nil
  127. }
  128. list := make([]Task, 0)
  129. err = e.Where("status = ? AND level = ?", Enabled, TaskLevelParent).
  130. In("id", taskIds...).
  131. Find(&list)
  132. if err != nil {
  133. return list, err
  134. }
  135. return t.setHostsForTasks(list)
  136. }
  137. func (t *Task) setHostsForTasks(tasks []Task) ([]Task, error) {
  138. taskHostModel := new(TaskHost)
  139. var err error
  140. for i, value := range tasks {
  141. taskHostDetails, err := taskHostModel.GetHostIdsByTaskId(value.Id)
  142. if err != nil {
  143. return nil, err
  144. }
  145. tasks[i].Hosts = taskHostDetails
  146. }
  147. return tasks, err
  148. }
  149. // 判断任务名称是否存在
  150. func (t *Task) NameExist(name string, id int) (bool, error) {
  151. e := db.Eloquent.Master()
  152. if id > 0 {
  153. count, err := e.Where("name = ? AND status = ? AND id != ?", name, Enabled, id).Count(t)
  154. return count > 0, err
  155. }
  156. count, err := e.Where("name = ? AND status = ?", name, Enabled).Count(t)
  157. return count > 0, err
  158. }
  159. func (t *Task) GetStatus(id int) (Status, error) {
  160. e := db.Eloquent.Master()
  161. exist, err := e.Id(id).Get(t)
  162. if err != nil {
  163. return 0, err
  164. }
  165. if !exist {
  166. return 0, errors.New("not exist")
  167. }
  168. return t.Status, nil
  169. }
  170. func (t *Task) Detail(id int) (Task, error) {
  171. result := Task{}
  172. e := db.Eloquent.Master()
  173. _, err := e.Where("id=?", id).Get(&result)
  174. if err != nil {
  175. return result, err
  176. }
  177. taskHostModel := new(TaskHost)
  178. result.Hosts, err = taskHostModel.GetHostIdsByTaskId(id)
  179. return result, err
  180. }
  181. func (task *Task) List(params CommonMap,pageSize int, pageIndex int) ([]Task, error) {
  182. e := db.Eloquent.Master()
  183. Offset := (pageIndex - 1) * pageSize
  184. list := make([]Task, 0)
  185. session := e.Alias("t").Join("LEFT", taskHostTableName(), "t.id = th.task_id")
  186. task.parseWhere(session, params)
  187. err := session.GroupBy("t.id").Desc("t.id").Cols("t.*").Limit(pageSize, Offset).Find(&list)
  188. if err != nil {
  189. return nil, err
  190. }
  191. return task.setHostsForTasks(list)
  192. }
  193. // 获取依赖任务列表
  194. func (task *Task) GetDependencyTaskList(ids string) ([]Task, error) {
  195. e := db.Eloquent.Master()
  196. list := make([]Task, 0)
  197. if ids == "" {
  198. return list, nil
  199. }
  200. idList := strings.Split(ids, ",")
  201. taskIds := make([]interface{}, len(idList))
  202. for i, v := range idList {
  203. taskIds[i] = v
  204. }
  205. fields := "t.*"
  206. err := e.Alias("t").
  207. Where("t.level = ?", TaskLevelChild).
  208. In("t.id", taskIds).
  209. Cols(fields).
  210. Find(&list)
  211. if err != nil {
  212. return list, err
  213. }
  214. return task.setHostsForTasks(list)
  215. }
  216. func (task *Task) Total(params CommonMap) (int64, error) {
  217. e := db.Eloquent.Master()
  218. session := e.Alias("t").Join("LEFT", taskHostTableName(), "t.id = th.task_id")
  219. task.parseWhere(session, params)
  220. list := make([]Task, 0)
  221. err := session.GroupBy("t.id").Find(&list)
  222. return int64(len(list)), err
  223. }
  224. // 解析where
  225. func (task *Task) parseWhere(session *xorm.Session, params CommonMap) {
  226. if len(params) == 0 {
  227. return
  228. }
  229. id, ok := params["Id"]
  230. if ok && id.(int) > 0 {
  231. session.And("t.id = ?", id)
  232. }
  233. hostId, ok := params["HostId"]
  234. if ok && hostId.(int) > 0 {
  235. session.And("th.host_id = ?", hostId)
  236. }
  237. name, ok := params["Name"]
  238. if ok && name.(string) != "" {
  239. session.And("t.name LIKE ?", "%"+name.(string)+"%")
  240. }
  241. protocol, ok := params["Protocol"]
  242. if ok && protocol.(int) > 0 {
  243. session.And("protocol = ?", protocol)
  244. }
  245. status, ok := params["Status"]
  246. if ok && status.(int) > -1 {
  247. session.And("status = ?", status)
  248. }
  249. tag, ok := params["Tag"]
  250. if ok && tag.(string) != "" {
  251. session.And("tag = ? ", tag)
  252. }
  253. }