2017-03-10 09:24:06 +00:00
|
|
|
|
package models
|
|
|
|
|
|
|
|
|
|
import (
|
2017-09-16 09:58:33 +00:00
|
|
|
|
"errors"
|
|
|
|
|
"github.com/go-xorm/xorm"
|
|
|
|
|
"strings"
|
|
|
|
|
"time"
|
2017-03-10 09:24:06 +00:00
|
|
|
|
)
|
|
|
|
|
|
2017-04-13 09:35:59 +00:00
|
|
|
|
type TaskProtocol int8
|
2017-03-10 09:24:06 +00:00
|
|
|
|
|
2017-03-24 05:06:53 +00:00
|
|
|
|
const (
|
2017-09-16 09:58:33 +00:00
|
|
|
|
TaskHTTP TaskProtocol = iota + 1 // HTTP协议
|
|
|
|
|
TaskRPC // RPC方式执行命令
|
2017-03-24 05:06:53 +00:00
|
|
|
|
)
|
|
|
|
|
|
2017-06-08 10:04:55 +00:00
|
|
|
|
type TaskLevel int8
|
|
|
|
|
|
|
|
|
|
const (
|
2017-09-16 09:58:33 +00:00
|
|
|
|
TaskLevelParent TaskLevel = 1 // 父任务
|
|
|
|
|
TaskLevelChild TaskLevel = 2 // 子任务(依赖任务)
|
2017-06-08 10:04:55 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type TaskDependencyStatus int8
|
|
|
|
|
|
|
|
|
|
const (
|
2017-09-16 09:58:33 +00:00
|
|
|
|
TaskDependencyStatusStrong TaskDependencyStatus = 1 // 强依赖
|
|
|
|
|
TaskDependencyStatusWeak TaskDependencyStatus = 2 // 弱依赖
|
2017-06-08 10:04:55 +00:00
|
|
|
|
)
|
|
|
|
|
|
2017-03-23 05:31:16 +00:00
|
|
|
|
// 任务
|
2017-03-10 09:24:06 +00:00
|
|
|
|
type Task struct {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
Id int `xorm:"int pk autoincr"`
|
|
|
|
|
Name string `xorm:"varchar(32) notnull"` // 任务名称
|
2017-12-09 03:01:48 +00:00
|
|
|
|
Level TaskLevel `xorm:"tinyint notnull index default 1"` // 任务等级 1: 主任务 2: 依赖任务
|
2017-09-16 09:58:33 +00:00
|
|
|
|
DependencyTaskId string `xorm:"varchar(64) notnull default ''"` // 依赖任务ID,多个ID逗号分隔
|
2017-12-09 03:01:48 +00:00
|
|
|
|
DependencyStatus TaskDependencyStatus `xorm:"tinyint notnull default 1"` // 依赖关系 1:强依赖 主任务执行成功, 依赖任务才会被执行 2:弱依赖
|
2017-09-16 09:58:33 +00:00
|
|
|
|
Spec string `xorm:"varchar(64) notnull"` // crontab
|
|
|
|
|
Protocol TaskProtocol `xorm:"tinyint notnull index"` // 协议 1:http 2:系统命令
|
|
|
|
|
Command string `xorm:"varchar(256) notnull"` // URL地址或shell命令
|
|
|
|
|
Timeout int `xorm:"mediumint notnull default 0"` // 任务执行超时时间(单位秒),0不限制
|
|
|
|
|
Multi int8 `xorm:"tinyint notnull default 1"` // 是否允许多实例运行
|
|
|
|
|
RetryTimes int8 `xorm:"tinyint notnull default 0"` // 重试次数
|
2017-12-09 03:01:48 +00:00
|
|
|
|
RetryInterval int16 `xorm:"smallint notnull default 0"` // 重试间隔时间
|
|
|
|
|
NotifyStatus int8 `xorm:"tinyint notnull default 1"` // 任务执行结束是否通知 0: 不通知 1: 失败通知 2: 执行结束通知
|
|
|
|
|
NotifyType int8 `xorm:"tinyint notnull default 0"` // 通知类型 1: 邮件 2: slack
|
2017-09-16 09:58:33 +00:00
|
|
|
|
NotifyReceiverId string `xorm:"varchar(256) notnull default '' "` // 通知接受者ID, setting表主键ID,多个ID逗号分隔
|
|
|
|
|
Tag string `xorm:"varchar(32) notnull default ''"`
|
|
|
|
|
Remark string `xorm:"varchar(100) notnull default ''"` // 备注
|
|
|
|
|
Status Status `xorm:"tinyint notnull index default 0"` // 状态 1:正常 0:停止
|
|
|
|
|
Created time.Time `xorm:"datetime notnull created"` // 创建时间
|
|
|
|
|
Deleted time.Time `xorm:"datetime deleted"` // 删除时间
|
|
|
|
|
BaseModel `xorm:"-"`
|
|
|
|
|
Hosts []TaskHostDetail `xorm:"-"`
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func taskHostTableName() []string {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return []string{TablePrefix + "task_host", "th"}
|
2017-04-10 09:37:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-10 09:24:06 +00:00
|
|
|
|
// 新增
|
2017-04-02 02:19:52 +00:00
|
|
|
|
func (task *Task) Create() (insertId int, err error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
_, err = Db.Insert(task)
|
|
|
|
|
if err == nil {
|
|
|
|
|
insertId = task.Id
|
|
|
|
|
}
|
2017-03-24 09:55:44 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-12 01:59:22 +00:00
|
|
|
|
// 新增测试任务
|
|
|
|
|
func (task *Task) CreateTestTask() {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
// HTTP任务
|
|
|
|
|
task.Name = "测试HTTP任务"
|
|
|
|
|
task.Level = TaskLevelParent
|
|
|
|
|
task.Protocol = TaskHTTP
|
|
|
|
|
task.Spec = "*/30 * * * * *"
|
|
|
|
|
task.Tag = "test-task"
|
|
|
|
|
// 查询IP地址区域信息
|
|
|
|
|
task.Command = "http://ip.taobao.com/service/getIpInfo.php?ip=117.27.140.253"
|
|
|
|
|
task.Status = Enabled
|
|
|
|
|
task.Create()
|
2017-05-12 01:59:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
func (task *Task) UpdateBean(id int) (int64, error) {
|
|
|
|
|
return Db.ID(id).
|
2017-12-09 03:01:48 +00:00
|
|
|
|
Cols("name,spec,protocol,command,timeout,multi," +
|
|
|
|
|
"retry_times,retry_interval,remark,notify_status," +
|
|
|
|
|
"notify_type,notify_receiver_id, dependency_task_id, dependency_status, tag").
|
2017-09-16 09:58:33 +00:00
|
|
|
|
Update(task)
|
2017-04-17 10:04:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-10 09:24:06 +00:00
|
|
|
|
// 更新
|
2017-04-02 02:19:52 +00:00
|
|
|
|
func (task *Task) Update(id int, data CommonMap) (int64, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return Db.Table(task).ID(id).Update(data)
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 删除
|
2017-04-02 02:19:52 +00:00
|
|
|
|
func (task *Task) Delete(id int) (int64, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return Db.Id(id).Delete(task)
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 禁用
|
2017-04-02 02:19:52 +00:00
|
|
|
|
func (task *Task) Disable(id int) (int64, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return task.Update(id, CommonMap{"status": Disabled})
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 激活
|
2017-04-02 02:19:52 +00:00
|
|
|
|
func (task *Task) Enable(id int) (int64, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return task.Update(id, CommonMap{"status": Enabled})
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-20 01:36:42 +00:00
|
|
|
|
// 获取所有激活任务
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func (task *Task) ActiveList() ([]Task, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
list := make([]Task, 0)
|
|
|
|
|
err := Db.Where("status = ? AND level = ?", Enabled, TaskLevelParent).
|
|
|
|
|
Find(&list)
|
2017-03-24 09:55:44 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
return list, err
|
|
|
|
|
}
|
2017-08-06 14:49:24 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return task.setHostsForTasks(list)
|
2017-03-24 09:55:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-20 01:36:42 +00:00
|
|
|
|
// 获取某个主机下的所有激活任务
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func (task *Task) ActiveListByHostId(hostId int16) ([]Task, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
taskHostModel := new(TaskHost)
|
|
|
|
|
taskIds, err := taskHostModel.GetTaskIdsByHostId(hostId)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
list := make([]Task, 0)
|
|
|
|
|
err = Db.Where("status = ? AND level = ?", Enabled, TaskLevelParent).
|
|
|
|
|
In("id", taskIds...).
|
|
|
|
|
Find(&list)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return list, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return task.setHostsForTasks(list)
|
2017-04-20 01:36:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func (task *Task) setHostsForTasks(tasks []Task) ([]Task, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
taskHostModel := new(TaskHost)
|
|
|
|
|
var err error
|
|
|
|
|
for i, value := range tasks {
|
|
|
|
|
taskHostDetails, err := taskHostModel.GetHostIdsByTaskId(value.Id)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
tasks[i].Hosts = taskHostDetails
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return tasks, err
|
2017-04-13 09:35:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 判断任务名称是否存在
|
2017-09-16 09:58:33 +00:00
|
|
|
|
func (task *Task) NameExist(name string, id int) (bool, error) {
|
|
|
|
|
if id > 0 {
|
|
|
|
|
count, err := Db.Where("name = ? AND status = ? AND id != ?", name, Enabled, id).Count(task)
|
|
|
|
|
return count > 0, err
|
|
|
|
|
}
|
|
|
|
|
count, err := Db.Where("name = ? AND status = ?", name, Enabled).Count(task)
|
|
|
|
|
|
|
|
|
|
return count > 0, err
|
2017-04-13 09:35:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-27 13:42:18 +00:00
|
|
|
|
func (task *Task) GetStatus(id int) (Status, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
exist, err := Db.Id(id).Get(task)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return 0, err
|
|
|
|
|
}
|
|
|
|
|
if !exist {
|
|
|
|
|
return 0, errors.New("not exist")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return task.Status, nil
|
2017-05-27 13:42:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
func (task *Task) Detail(id int) (Task, error) {
|
|
|
|
|
t := Task{}
|
|
|
|
|
_, err := Db.Where("id=?", id).Get(&t)
|
2017-08-06 14:49:24 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
return t, err
|
|
|
|
|
}
|
2017-08-06 14:49:24 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
taskHostModel := new(TaskHost)
|
|
|
|
|
t.Hosts, err = taskHostModel.GetHostIdsByTaskId(id)
|
2017-04-10 09:37:16 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return t, err
|
2017-04-10 09:37:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func (task *Task) List(params CommonMap) ([]Task, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
task.parsePageAndPageSize(params)
|
|
|
|
|
list := make([]Task, 0)
|
|
|
|
|
session := Db.Alias("t").Join("LEFT", taskHostTableName(), "t.id = th.task_id")
|
|
|
|
|
task.parseWhere(session, params)
|
|
|
|
|
err := session.GroupBy("t.id").Desc("t.id").Cols("t.*").Limit(task.PageSize, task.pageLimitOffset()).Find(&list)
|
2017-03-10 09:24:06 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2017-08-06 14:49:24 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return task.setHostsForTasks(list)
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
2017-06-08 10:04:55 +00:00
|
|
|
|
|
|
|
|
|
// 获取依赖任务列表
|
2017-08-06 14:49:24 +00:00
|
|
|
|
func (task *Task) GetDependencyTaskList(ids string) ([]Task, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
list := make([]Task, 0)
|
|
|
|
|
if ids == "" {
|
|
|
|
|
return list, nil
|
|
|
|
|
}
|
|
|
|
|
idList := strings.Split(ids, ",")
|
|
|
|
|
taskIds := make([]interface{}, len(idList))
|
|
|
|
|
for i, v := range idList {
|
|
|
|
|
taskIds[i] = v
|
|
|
|
|
}
|
|
|
|
|
fields := "t.*"
|
|
|
|
|
err := Db.Alias("t").
|
|
|
|
|
Where("t.level = ?", TaskLevelChild).
|
|
|
|
|
In("t.id", taskIds).
|
|
|
|
|
Cols(fields).
|
|
|
|
|
Find(&list)
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
return list, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return task.setHostsForTasks(list)
|
2017-06-08 10:04:55 +00:00
|
|
|
|
}
|
2017-03-10 09:24:06 +00:00
|
|
|
|
|
2017-04-23 06:11:53 +00:00
|
|
|
|
func (task *Task) Total(params CommonMap) (int64, error) {
|
2017-09-16 09:58:33 +00:00
|
|
|
|
session := Db.Alias("t").Join("LEFT", taskHostTableName(), "t.id = th.task_id")
|
|
|
|
|
task.parseWhere(session, params)
|
|
|
|
|
list := make([]Task, 0)
|
2017-09-06 01:43:13 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
err := session.GroupBy("t.id").Find(&list)
|
2017-09-06 01:43:13 +00:00
|
|
|
|
|
2017-09-16 09:58:33 +00:00
|
|
|
|
return int64(len(list)), err
|
2017-03-10 09:24:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-21 09:41:59 +00:00
|
|
|
|
// 解析where
|
2017-09-16 09:58:33 +00:00
|
|
|
|
func (task *Task) parseWhere(session *xorm.Session, params CommonMap) {
|
|
|
|
|
if len(params) == 0 {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
id, ok := params["Id"]
|
|
|
|
|
if ok && id.(int) > 0 {
|
|
|
|
|
session.And("t.id = ?", id)
|
|
|
|
|
}
|
|
|
|
|
hostId, ok := params["HostId"]
|
|
|
|
|
if ok && hostId.(int) > 0 {
|
|
|
|
|
session.And("th.host_id = ?", hostId)
|
|
|
|
|
}
|
|
|
|
|
name, ok := params["Name"]
|
|
|
|
|
if ok && name.(string) != "" {
|
|
|
|
|
session.And("t.name LIKE ?", "%"+name.(string)+"%")
|
|
|
|
|
}
|
|
|
|
|
protocol, ok := params["Protocol"]
|
|
|
|
|
if ok && protocol.(int) > 0 {
|
|
|
|
|
session.And("protocol = ?", protocol)
|
|
|
|
|
}
|
|
|
|
|
status, ok := params["Status"]
|
|
|
|
|
if ok && status.(int) > -1 {
|
|
|
|
|
session.And("status = ?", status)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tag, ok := params["Tag"]
|
|
|
|
|
if ok && tag.(string) != "" {
|
|
|
|
|
session.And("tag = ? ", tag)
|
|
|
|
|
}
|
2017-04-26 08:17:18 +00:00
|
|
|
|
}
|