mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-06-05 19:26:45 +08:00
961 lines
28 KiB
Go
961 lines
28 KiB
Go
// Copyright 2019 Yunion
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package taskman
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
"net/http"
|
|
"reflect"
|
|
"runtime/debug"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/appctx"
|
|
"yunion.io/x/pkg/errors"
|
|
"yunion.io/x/pkg/gotypes"
|
|
"yunion.io/x/pkg/util/httputils"
|
|
"yunion.io/x/pkg/util/rbacscope"
|
|
"yunion.io/x/pkg/util/reflectutils"
|
|
"yunion.io/x/pkg/util/stringutils"
|
|
"yunion.io/x/pkg/util/timeutils"
|
|
"yunion.io/x/pkg/util/version"
|
|
"yunion.io/x/pkg/utils"
|
|
"yunion.io/x/sqlchemy"
|
|
|
|
"yunion.io/x/onecloud/pkg/apis"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db/quotas"
|
|
"yunion.io/x/onecloud/pkg/httperrors"
|
|
"yunion.io/x/onecloud/pkg/mcclient"
|
|
"yunion.io/x/onecloud/pkg/mcclient/auth"
|
|
"yunion.io/x/onecloud/pkg/mcclient/modules/yunionconf"
|
|
"yunion.io/x/onecloud/pkg/util/logclient"
|
|
)
|
|
|
|
const (
|
|
PARENT_TASK_ID_KEY = "parent_task_id"
|
|
PENDING_USAGE_KEY = "__pending_usage__"
|
|
PARENT_TASK_NOTIFY_KEY = "__parent_task_notifyurl"
|
|
REQUEST_CONTEXT_KEY = "__request_context"
|
|
|
|
TASK_STAGE_FAILED = "failed"
|
|
TASK_STAGE_COMPLETE = "complete"
|
|
|
|
MAX_REMOTE_NOTIFY_TRIES = 5
|
|
|
|
MULTI_OBJECTS_ID = "[--MULTI_OBJECTS--]"
|
|
|
|
TASK_INIT_STAGE = "on_init"
|
|
|
|
CONVERT_TASK = "convert_task"
|
|
|
|
LANG = "lang"
|
|
)
|
|
|
|
type STaskManager struct {
|
|
db.SResourceBaseManager
|
|
}
|
|
|
|
var TaskManager *STaskManager
|
|
var userCredWidthLimit = 0
|
|
|
|
func init() {
|
|
TaskManager = &STaskManager{
|
|
SResourceBaseManager: db.NewResourceBaseManager(STask{}, "tasks_tbl", "task", "tasks")}
|
|
TaskManager.SetVirtualObject(TaskManager)
|
|
if field, ok := reflect.TypeOf(&STask{}).Elem().FieldByName("UserCred"); ok {
|
|
if widthStr := field.Tag.Get(sqlchemy.TAG_WIDTH); len(widthStr) > 0 {
|
|
userCredWidthLimit, _ = strconv.Atoi(widthStr)
|
|
}
|
|
}
|
|
}
|
|
|
|
type STask struct {
|
|
db.SResourceBase
|
|
|
|
Id string `width:"36" charset:"ascii" primary:"true" list:"user"` // Column(VARCHAR(36, charset='ascii'), primary_key=True, default=get_uuid)
|
|
|
|
ObjName string `width:"128" charset:"utf8" nullable:"false" list:"user"` // Column(VARCHAR(128, charset='utf8'), nullable=False)
|
|
ObjId string `width:"128" charset:"ascii" nullable:"false" list:"user" index:"true"` // Column(VARCHAR(ID_LENGTH, charset='ascii'), nullable=False)
|
|
TaskName string `width:"64" charset:"ascii" nullable:"false" list:"user"` // Column(VARCHAR(64, charset='ascii'), nullable=False)
|
|
UserCred mcclient.TokenCredential `width:"1024" charset:"utf8" nullable:"false" get:"user"` // Column(VARCHAR(1024, charset='ascii'), nullable=False)
|
|
// OwnerCred string `width:"512" charset:"ascii" nullable:"true"` // Column(VARCHAR(512, charset='ascii'), nullable=True)
|
|
Params *jsonutils.JSONDict `charset:"utf8" length:"medium" nullable:"false" get:"user"` // Column(MEDIUMTEXT(charset='ascii'), nullable=False)
|
|
|
|
Stage string `width:"64" charset:"ascii" nullable:"false" default:"on_init" list:"user"` // Column(VARCHAR(64, charset='ascii'), nullable=False, default='on_init')
|
|
|
|
taskObject db.IStandaloneModel `ignore:"true"`
|
|
taskObjects []db.IStandaloneModel `ignore:"true"`
|
|
}
|
|
|
|
func (manager *STaskManager) CreateByInsertOrUpdate() bool {
|
|
return false
|
|
}
|
|
|
|
func (manager *STaskManager) AllowListItems(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject) bool {
|
|
return true
|
|
}
|
|
|
|
func (manager *STaskManager) AllowCreateItem(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) bool {
|
|
return false
|
|
}
|
|
|
|
func (manager *STaskManager) FilterById(q *sqlchemy.SQuery, idStr string) *sqlchemy.SQuery {
|
|
return q.Equals("id", idStr)
|
|
}
|
|
|
|
func (manager *STaskManager) FilterByNotId(q *sqlchemy.SQuery, idStr string) *sqlchemy.SQuery {
|
|
return q.NotEquals("id", idStr)
|
|
}
|
|
|
|
func (manager *STaskManager) FilterByName(q *sqlchemy.SQuery, name string) *sqlchemy.SQuery {
|
|
return q
|
|
}
|
|
|
|
func (manager *STaskManager) PerformAction(ctx context.Context, userCred mcclient.TokenCredential, taskId string, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
|
|
err := runTask(taskId, data)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "runTask")
|
|
}
|
|
resp := jsonutils.NewDict()
|
|
// 'result': 'ok'
|
|
resp.Add(jsonutils.NewString("ok"), "result")
|
|
return resp, nil
|
|
}
|
|
|
|
func (manager *STask) PreCheckPerformAction(
|
|
ctx context.Context, userCred mcclient.TokenCredential,
|
|
action string, query jsonutils.JSONObject, data jsonutils.JSONObject,
|
|
) error {
|
|
return nil
|
|
}
|
|
|
|
func (self *STask) GetOwnerId() mcclient.IIdentityProvider {
|
|
owner := db.SOwnerId{DomainId: self.UserCred.GetProjectDomainId(), Domain: self.UserCred.GetProjectDomain(),
|
|
ProjectId: self.UserCred.GetProjectId(), Project: self.UserCred.GetProjectName()}
|
|
return &owner
|
|
}
|
|
|
|
func (manager *STaskManager) FilterByOwner(ctx context.Context, q *sqlchemy.SQuery, man db.FilterByOwnerProvider, userCred mcclient.TokenCredential, owner mcclient.IIdentityProvider, scope rbacscope.TRbacScope) *sqlchemy.SQuery {
|
|
if owner != nil {
|
|
switch scope {
|
|
case rbacscope.ScopeProject:
|
|
if len(owner.GetProjectId()) > 0 {
|
|
q = q.Contains("user_cred", owner.GetProjectId())
|
|
}
|
|
case rbacscope.ScopeDomain:
|
|
if len(owner.GetProjectDomainId()) > 0 {
|
|
q = q.Contains("user_cred", owner.GetProjectDomainId())
|
|
}
|
|
}
|
|
}
|
|
return q
|
|
}
|
|
|
|
func (manager *STaskManager) FetchTaskById(taskId string) *STask {
|
|
return manager.fetchTask(taskId)
|
|
}
|
|
|
|
func (self *STask) AllowUpdateItem(ctx context.Context, userCred mcclient.TokenCredential) bool {
|
|
return false
|
|
}
|
|
|
|
func (self *STask) AllowDeleteItem(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) bool {
|
|
return false
|
|
}
|
|
|
|
func (self *STask) ValidateDeleteCondition(ctx context.Context, info jsonutils.JSONObject) error {
|
|
return httperrors.NewForbiddenError("forbidden")
|
|
}
|
|
|
|
func (self *STask) ValidateUpdateCondition(ctx context.Context) error {
|
|
return httperrors.NewForbiddenError("forbidden")
|
|
}
|
|
|
|
func (self *STask) BeforeInsert() {
|
|
if len(self.Id) == 0 {
|
|
self.Id = stringutils.UUID4()
|
|
}
|
|
}
|
|
|
|
func (self *STask) GetId() string {
|
|
return self.Id
|
|
}
|
|
|
|
func (self *STask) GetName() string {
|
|
return self.TaskName
|
|
}
|
|
|
|
func fetchTaskParams(
|
|
ctx context.Context,
|
|
taskName string,
|
|
taskData *jsonutils.JSONDict,
|
|
parentTaskId string,
|
|
parentTaskNotifyUrl string,
|
|
pendingUsages []quotas.IQuota,
|
|
) *jsonutils.JSONDict {
|
|
var data *jsonutils.JSONDict
|
|
if taskData != nil {
|
|
excludeKeys := []string{
|
|
PARENT_TASK_ID_KEY, PARENT_TASK_NOTIFY_KEY, PENDING_USAGE_KEY,
|
|
}
|
|
for i := 1; taskData.Contains(pendingUsageKey(i)); i += 1 {
|
|
excludeKeys = append(excludeKeys, pendingUsageKey(i))
|
|
}
|
|
data = taskData.CopyExcludes(excludeKeys...)
|
|
} else {
|
|
data = jsonutils.NewDict()
|
|
}
|
|
reqContext := appctx.FetchAppContextData(ctx)
|
|
if !reqContext.IsZero() {
|
|
data.Add(jsonutils.Marshal(&reqContext), REQUEST_CONTEXT_KEY)
|
|
}
|
|
if len(parentTaskId) > 0 || len(parentTaskNotifyUrl) > 0 {
|
|
if len(parentTaskId) > 0 {
|
|
data.Add(jsonutils.NewString(parentTaskId), PARENT_TASK_ID_KEY)
|
|
}
|
|
if len(parentTaskNotifyUrl) > 0 {
|
|
data.Add(jsonutils.NewString(parentTaskNotifyUrl), PARENT_TASK_NOTIFY_KEY)
|
|
log.Infof("%s notify parent url: %s", taskName, parentTaskNotifyUrl)
|
|
}
|
|
} else {
|
|
if !reqContext.IsZero() {
|
|
if len(reqContext.TaskId) > 0 && len(reqContext.TaskNotifyUrl) == 0 {
|
|
data.Add(jsonutils.NewString(reqContext.TaskId), PARENT_TASK_ID_KEY)
|
|
}
|
|
if len(reqContext.TaskNotifyUrl) > 0 {
|
|
data.Add(jsonutils.NewString(reqContext.TaskNotifyUrl), PARENT_TASK_NOTIFY_KEY)
|
|
log.Infof("%s notify parent url: %s", taskName, reqContext.TaskNotifyUrl)
|
|
}
|
|
}
|
|
}
|
|
if len(pendingUsages) > 0 {
|
|
for i := range pendingUsages {
|
|
pendingUsage := pendingUsages[i]
|
|
if gotypes.IsNil(pendingUsage) {
|
|
continue
|
|
}
|
|
key := pendingUsageKey(i)
|
|
data.Add(jsonutils.Marshal(pendingUsage), key)
|
|
}
|
|
}
|
|
return data
|
|
}
|
|
|
|
func (manager *STaskManager) NewTask(
|
|
ctx context.Context,
|
|
taskName string,
|
|
obj db.IStandaloneModel,
|
|
userCred mcclient.TokenCredential,
|
|
taskData *jsonutils.JSONDict,
|
|
parentTaskId string,
|
|
parentTaskNotifyUrl string,
|
|
pendingUsage ...quotas.IQuota,
|
|
) (*STask, error) {
|
|
if !isTaskExist(taskName) {
|
|
return nil, fmt.Errorf("task %s not found", taskName)
|
|
}
|
|
|
|
data := fetchTaskParams(ctx, taskName, taskData, parentTaskId, parentTaskNotifyUrl, pendingUsage)
|
|
task := &STask{
|
|
ObjName: obj.Keyword(),
|
|
ObjId: obj.GetId(),
|
|
TaskName: taskName,
|
|
UserCred: userCred,
|
|
Params: data,
|
|
Stage: TASK_INIT_STAGE,
|
|
}
|
|
|
|
if userCredWidthLimit > 0 && len(userCred.String()) > userCredWidthLimit {
|
|
return nil, fmt.Errorf("Too many permissions for user %s", userCred.GetUserName())
|
|
}
|
|
|
|
task.SetModelManager(manager, task)
|
|
err := manager.TableSpec().Insert(ctx, task)
|
|
if err != nil {
|
|
log.Errorf("Task insert error %s", err)
|
|
return nil, err
|
|
}
|
|
parentTask := task.GetParentTask()
|
|
if parentTask != nil {
|
|
st := &SSubTask{TaskId: parentTask.Id, Stage: parentTask.Stage, SubtaskId: task.Id}
|
|
st.SetModelManager(SubTaskManager, st)
|
|
err := SubTaskManager.TableSpec().Insert(ctx, st)
|
|
if err != nil {
|
|
log.Errorf("Subtask insert error %s", err)
|
|
return nil, err
|
|
}
|
|
}
|
|
return task, nil
|
|
}
|
|
|
|
func (manager *STaskManager) NewParallelTask(
|
|
ctx context.Context,
|
|
taskName string,
|
|
objs []db.IStandaloneModel,
|
|
userCred mcclient.TokenCredential,
|
|
taskData *jsonutils.JSONDict,
|
|
parentTaskId string,
|
|
parentTaskNotifyUrl string,
|
|
pendingUsage ...quotas.IQuota,
|
|
) (*STask, error) {
|
|
if !isTaskExist(taskName) {
|
|
return nil, fmt.Errorf("task %s not found", taskName)
|
|
}
|
|
|
|
if len(objs) == 0 {
|
|
return nil, fmt.Errorf("failed to do task %s with zero objs", taskName)
|
|
}
|
|
|
|
log.Debugf("number of objs: %d", len(objs))
|
|
|
|
data := fetchTaskParams(ctx, taskName, taskData, parentTaskId, parentTaskNotifyUrl, pendingUsage)
|
|
task := &STask{
|
|
ObjName: objs[0].Keyword(),
|
|
ObjId: MULTI_OBJECTS_ID,
|
|
TaskName: taskName,
|
|
UserCred: userCred,
|
|
Params: data,
|
|
Stage: TASK_INIT_STAGE,
|
|
}
|
|
task.SetModelManager(manager, task)
|
|
err := manager.TableSpec().Insert(ctx, task)
|
|
if err != nil {
|
|
log.Errorf("Task insert error %s", err)
|
|
return nil, err
|
|
}
|
|
|
|
for _, obj := range objs {
|
|
to := STaskObject{TaskId: task.Id, ObjId: obj.GetId()}
|
|
to.SetModelManager(TaskObjectManager, &to)
|
|
err := TaskObjectManager.TableSpec().Insert(ctx, &to)
|
|
if err != nil {
|
|
log.Errorf("Taskobject insert error %s", err)
|
|
return nil, errors.Wrap(err, "insert task object")
|
|
}
|
|
}
|
|
|
|
parentTask := task.GetParentTask()
|
|
if parentTask != nil {
|
|
st := SSubTask{TaskId: parentTask.Id, Stage: parentTask.Stage, SubtaskId: task.Id}
|
|
err := SubTaskManager.TableSpec().Insert(ctx, &st)
|
|
if err != nil {
|
|
log.Errorf("Subtask insert error %s", err)
|
|
return nil, err
|
|
}
|
|
}
|
|
return task, nil
|
|
}
|
|
|
|
func (manager *STaskManager) fetchTask(idStr string) *STask {
|
|
iTask, err := db.NewModelObject(manager)
|
|
if err != nil {
|
|
log.Errorf("New task object fail: %s", err)
|
|
return nil
|
|
}
|
|
err = manager.Query().Equals("id", idStr).First(iTask)
|
|
if err != nil {
|
|
log.Errorf("GetTask %s fail: %s", idStr, err)
|
|
return nil
|
|
}
|
|
task := iTask.(*STask)
|
|
if task.Params == nil {
|
|
task.Params = jsonutils.NewDict()
|
|
}
|
|
return task
|
|
}
|
|
|
|
func (manager *STaskManager) getTaskName(taskId string) string {
|
|
baseTask := manager.fetchTask(taskId)
|
|
if baseTask == nil {
|
|
return ""
|
|
}
|
|
return baseTask.TaskName
|
|
}
|
|
|
|
func (manager *STaskManager) execTask(taskId string, data jsonutils.JSONObject) {
|
|
baseTask := manager.fetchTask(taskId)
|
|
if baseTask == nil {
|
|
return
|
|
}
|
|
taskType, ok := taskTable[baseTask.TaskName]
|
|
if !ok {
|
|
log.Errorf("Cannot find task %s", baseTask.TaskName)
|
|
return
|
|
}
|
|
log.Debugf("Do task %s(%s) with data %s at stage %s", taskType, taskId, data, baseTask.Stage)
|
|
taskValue := reflect.New(taskType)
|
|
if taskValue.Type().Implements(ITaskType) {
|
|
execITask(taskValue, baseTask, data, false)
|
|
} else if taskValue.Type().Implements(IBatchTaskType) {
|
|
execITask(taskValue, baseTask, data, true)
|
|
} else {
|
|
log.Errorf("Unsupported task type?? %s", taskValue.Type())
|
|
}
|
|
}
|
|
|
|
type sSortedObjects []db.IStandaloneModel
|
|
|
|
func (a sSortedObjects) Len() int { return len(a) }
|
|
func (a sSortedObjects) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
func (a sSortedObjects) Less(i, j int) bool { return a[i].GetId() < a[j].GetId() }
|
|
|
|
func execITask(taskValue reflect.Value, task *STask, odata jsonutils.JSONObject, isMulti bool) {
|
|
ctxData := task.GetRequestContext()
|
|
ctx := ctxData.GetContext()
|
|
|
|
taskFailed := false
|
|
|
|
var data jsonutils.JSONObject
|
|
if odata != nil {
|
|
switch dictdata := odata.(type) {
|
|
case *jsonutils.JSONDict:
|
|
taskStatus, _ := odata.GetString("__status__")
|
|
if len(taskStatus) > 0 && taskStatus != "OK" {
|
|
taskFailed = true
|
|
dictdata.Set("__stage__", jsonutils.NewString(task.Stage))
|
|
if !dictdata.Contains("__reason__") {
|
|
reasonJson := dictdata.CopyExcludes("__status__", "__stage__")
|
|
dictdata.Set("__reason__", reasonJson)
|
|
}
|
|
/*if vdata, ok := data.(*jsonutils.JSONDict); ok {
|
|
reason, err := vdata.Get("__reason__") // only dict support Get
|
|
if err != nil {
|
|
reason = jsonutils.NewString(fmt.Sprintf("Task failed due to unknown remote errors! %s", odata))
|
|
vdata.Set("__reason__", reason)
|
|
}
|
|
}*/
|
|
}
|
|
data = dictdata
|
|
default:
|
|
data = odata
|
|
}
|
|
} else {
|
|
data = jsonutils.NewDict()
|
|
}
|
|
|
|
stageName := task.Stage
|
|
if taskFailed {
|
|
stageName = fmt.Sprintf("%sFailed", task.Stage)
|
|
if strings.Contains(stageName, "_") {
|
|
stageName = fmt.Sprintf("%s_failed", task.Stage)
|
|
}
|
|
}
|
|
|
|
if strings.Contains(stageName, "_") {
|
|
stageName = utils.Kebab2Camel(stageName, "_")
|
|
}
|
|
|
|
funcValue := taskValue.MethodByName(stageName)
|
|
|
|
if !funcValue.IsValid() || funcValue.IsNil() {
|
|
msg := fmt.Sprintf("Stage %s not found", stageName)
|
|
if taskFailed {
|
|
// failed handler is optional, ignore the error
|
|
log.Warningf(msg)
|
|
msg, _ = data.GetString()
|
|
} else {
|
|
log.Errorf(msg)
|
|
}
|
|
task.SetStageFailed(ctx, jsonutils.NewString(msg))
|
|
task.SaveRequestContext(&ctxData)
|
|
return
|
|
}
|
|
|
|
objManager := db.GetModelManager(task.ObjName)
|
|
if objManager == nil {
|
|
msg := fmt.Sprintf("model %s not found??? ...", task.ObjName)
|
|
log.Errorf(msg)
|
|
task.SetStageFailed(ctx, jsonutils.NewString(msg))
|
|
task.SaveRequestContext(&ctxData)
|
|
return
|
|
}
|
|
// log.Debugf("objManager: %s", objManager)
|
|
objResManager, ok := objManager.(db.IStandaloneModelManager)
|
|
if !ok {
|
|
msg := fmt.Sprintf("model %s is not a resource??? ...", task.ObjName)
|
|
log.Errorf(msg)
|
|
task.SetStageFailed(ctx, jsonutils.NewString(msg))
|
|
task.SaveRequestContext(&ctxData)
|
|
return
|
|
}
|
|
|
|
params := make([]reflect.Value, 3)
|
|
params[0] = reflect.ValueOf(ctx)
|
|
|
|
if isMulti {
|
|
objIds := TaskObjectManager.GetObjectIds(task)
|
|
objs := make([]db.IStandaloneModel, len(objIds))
|
|
for i, objId := range objIds {
|
|
obj, err := objResManager.FetchById(objId)
|
|
if err != nil {
|
|
msg := fmt.Sprintf("fail to find %s object %s", task.ObjName, objId)
|
|
log.Errorf(msg)
|
|
task.SetStageFailed(ctx, jsonutils.NewString(msg))
|
|
task.SaveRequestContext(&ctxData)
|
|
return
|
|
}
|
|
objs[i] = obj.(db.IStandaloneModel)
|
|
}
|
|
task.taskObjects = objs
|
|
|
|
// sort objects by ids to avoid deadlock
|
|
sort.Sort(sSortedObjects(objs))
|
|
|
|
for i := range objs {
|
|
lockman.LockObject(ctx, objs[i])
|
|
defer lockman.ReleaseObject(ctx, objs[i])
|
|
}
|
|
|
|
params[1] = reflect.ValueOf(objs)
|
|
} else {
|
|
obj, err := objResManager.FetchById(task.ObjId)
|
|
if err != nil {
|
|
msg := fmt.Sprintf("fail to find %s object %s", task.ObjName, task.ObjId)
|
|
log.Errorf(msg)
|
|
task.SetStageFailed(ctx, jsonutils.NewString(msg))
|
|
task.SaveRequestContext(&ctxData)
|
|
return
|
|
}
|
|
task.taskObject = obj.(db.IStandaloneModel)
|
|
|
|
lockman.LockObject(ctx, obj)
|
|
defer lockman.ReleaseObject(ctx, obj)
|
|
|
|
params[1] = reflect.ValueOf(obj)
|
|
}
|
|
|
|
params[2] = reflect.ValueOf(data)
|
|
|
|
filled := reflectutils.FillEmbededStructValue(taskValue.Elem(), reflect.Indirect(reflect.ValueOf(task)))
|
|
if !filled {
|
|
log.Errorf("Cannot locate baseTask embedded struct, give up...")
|
|
return
|
|
}
|
|
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
// call set stage failed, should not call task.SetStageFailed
|
|
// func SetStageFailed may be overloading
|
|
yunionconf.BugReport.SendBugReport(ctx, version.GetShortString(), string(debug.Stack()), errors.Errorf("%s", r))
|
|
log.Errorf("Task %s PANIC on stage %s: %v \n%s", task.TaskName, stageName, r, debug.Stack())
|
|
SetStageFailedFuncValue := taskValue.MethodByName("SetStageFailed")
|
|
SetStageFailedFuncValue.Call(
|
|
[]reflect.Value{
|
|
reflect.ValueOf(ctx),
|
|
reflect.ValueOf(jsonutils.NewString(fmt.Sprintf("%v", r))),
|
|
},
|
|
)
|
|
obj, err := objResManager.FetchById(task.ObjId)
|
|
if err != nil {
|
|
return
|
|
}
|
|
statusObj, ok := obj.(db.IStatusStandaloneModel)
|
|
if ok {
|
|
db.StatusBaseSetStatus(ctx, statusObj, task.GetUserCred(), apis.STATUS_UNKNOWN, fmt.Sprintf("%v", r))
|
|
}
|
|
notes := map[string]interface{}{
|
|
"Stack": string(debug.Stack()),
|
|
"Version": version.GetShortString(),
|
|
"Task": task.TaskName,
|
|
"Stage": stageName,
|
|
"Message": fmt.Sprintf("%v", r),
|
|
}
|
|
logclient.AddSimpleActionLog(obj, logclient.ACT_PANIC, notes, task.GetUserCred(), false)
|
|
}
|
|
}()
|
|
|
|
log.Debugf("Call %s %s %#v", task.TaskName, stageName, params)
|
|
funcValue.Call(params)
|
|
|
|
// call save request context
|
|
saveRequestContextFuncValue := taskValue.MethodByName("SaveRequestContext")
|
|
saveRequestContextFuncValue.Call([]reflect.Value{reflect.ValueOf(&ctxData)})
|
|
}
|
|
|
|
func (task *STask) ScheduleRun(data jsonutils.JSONObject) error {
|
|
return runTask(task.Id, data)
|
|
}
|
|
|
|
func (self *STask) IsSubtask() bool {
|
|
return self.HasParentTask()
|
|
}
|
|
|
|
func (self *STask) HasParentTask() bool {
|
|
parentTaskId, _ := self.Params.GetString(PARENT_TASK_ID_KEY)
|
|
if len(parentTaskId) > 0 {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (self *STask) GetParentTask() *STask {
|
|
parentTaskId, _ := self.Params.GetString(PARENT_TASK_ID_KEY)
|
|
if len(parentTaskId) > 0 {
|
|
return TaskManager.fetchTask(parentTaskId)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (self *STask) GetRequestContext() appctx.AppContextData {
|
|
ctxData := appctx.AppContextData{}
|
|
if self.Params != nil {
|
|
ctxJson, _ := self.Params.Get(REQUEST_CONTEXT_KEY)
|
|
if ctxJson != nil {
|
|
ctxJson.Unmarshal(&ctxData)
|
|
}
|
|
}
|
|
return ctxData
|
|
}
|
|
|
|
func (self *STask) SaveRequestContext(data *appctx.AppContextData) {
|
|
jsonData := jsonutils.Marshal(data)
|
|
log.Debugf("SaveRequestContext %s param %s", jsonData, self.Params)
|
|
_, err := db.Update(self, func() error {
|
|
params := self.Params.CopyExcludes(REQUEST_CONTEXT_KEY)
|
|
params.Add(jsonData, REQUEST_CONTEXT_KEY)
|
|
self.Params = params
|
|
return nil
|
|
})
|
|
log.Debugf("Params: %s", self.Params)
|
|
if err != nil {
|
|
log.Errorf("save_request_context fail %s", err)
|
|
}
|
|
}
|
|
|
|
func (self *STask) SaveParams(data *jsonutils.JSONDict) error {
|
|
return self.SetStage("", data)
|
|
}
|
|
|
|
func (self *STask) SetStage(stageName string, data *jsonutils.JSONDict) error {
|
|
_, err := db.Update(self, func() error {
|
|
params := jsonutils.NewDict()
|
|
params.Update(self.Params)
|
|
if data != nil {
|
|
params.Update(data)
|
|
}
|
|
if len(stageName) > 0 {
|
|
stages, _ := params.Get("__stages")
|
|
if stages == nil {
|
|
stages = jsonutils.NewArray()
|
|
params.Add(stages, "__stages")
|
|
}
|
|
stageList := stages.(*jsonutils.JSONArray)
|
|
stageData := jsonutils.NewDict()
|
|
stageData.Add(jsonutils.NewString(self.Stage), "name")
|
|
stageData.Add(jsonutils.NewTimeString(time.Now()), "complete_at")
|
|
stageList.Add(stageData)
|
|
self.Stage = stageName
|
|
}
|
|
self.Params = params
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
log.Errorf("set_stage fail %s", err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (self *STask) GetObjectIdStr() string {
|
|
if self.ObjId == MULTI_OBJECTS_ID {
|
|
return strings.Join(TaskObjectManager.GetObjectIds(self), ",")
|
|
} else {
|
|
return self.ObjId
|
|
}
|
|
}
|
|
|
|
func (self *STask) SetStageComplete(ctx context.Context, data *jsonutils.JSONDict) {
|
|
log.Infof("XXX TASK %s complete", self.TaskName)
|
|
self.SetStage(TASK_STAGE_COMPLETE, data)
|
|
if data == nil {
|
|
data = jsonutils.NewDict()
|
|
}
|
|
if data.Size() == 0 {
|
|
data.Add(jsonutils.NewString(self.GetObjectIdStr()), "id")
|
|
data.Add(jsonutils.NewString(self.ObjName), "name")
|
|
}
|
|
self.NotifyParentTaskComplete(ctx, data, false)
|
|
}
|
|
|
|
func (self *STask) SetStageFailed(ctx context.Context, reason jsonutils.JSONObject) {
|
|
if self.Stage == TASK_STAGE_FAILED {
|
|
log.Warningf("Task %s has been failed", self.TaskName)
|
|
return
|
|
}
|
|
log.Infof("XXX TASK %s failed: %s on stage %s", self.TaskName, reason, self.Stage)
|
|
reasonDict := jsonutils.NewDict()
|
|
reasonDict.Add(jsonutils.NewString(self.Stage), "stage")
|
|
if reason != nil {
|
|
reasonDict.Add(reason, "reason")
|
|
}
|
|
reason = reasonDict
|
|
prevFailed, _ := self.Params.Get("__failed_reason")
|
|
if prevFailed != nil {
|
|
switch prevFailed.(type) {
|
|
case *jsonutils.JSONArray:
|
|
prevFailed.(*jsonutils.JSONArray).Add(reason)
|
|
reason = prevFailed
|
|
default:
|
|
reason = jsonutils.NewArray(prevFailed, reason)
|
|
}
|
|
}
|
|
data := jsonutils.NewDict()
|
|
data.Add(reason, "__failed_reason")
|
|
self.SetStage(TASK_STAGE_FAILED, data)
|
|
self.NotifyParentTaskFailure(ctx, reason)
|
|
}
|
|
|
|
func (self *STask) NotifyParentTaskComplete(ctx context.Context, body *jsonutils.JSONDict, failed bool) {
|
|
log.Infof("notify_parent_task_complete: %s params %s", self.TaskName, self.Params)
|
|
parentTaskId, _ := self.Params.GetString(PARENT_TASK_ID_KEY)
|
|
parentTaskNotify, _ := self.Params.GetString(PARENT_TASK_NOTIFY_KEY)
|
|
if len(parentTaskId) > 0 {
|
|
subTask := SubTaskManager.GetSubTask(parentTaskId, self.Id)
|
|
if subTask != nil {
|
|
subTask.SaveResults(failed, body)
|
|
}
|
|
func() {
|
|
lockman.LockRawObject(ctx, "tasks", parentTaskId)
|
|
defer lockman.ReleaseRawObject(ctx, "tasks", parentTaskId)
|
|
|
|
pTask := TaskManager.fetchTask(parentTaskId)
|
|
if pTask == nil {
|
|
log.Errorf("Parent task %s not found", parentTaskId)
|
|
return
|
|
}
|
|
if pTask.IsCurrentStageComplete() {
|
|
pTask.ScheduleRun(body)
|
|
}
|
|
}()
|
|
}
|
|
if len(parentTaskNotify) > 0 {
|
|
notifyRemoteTask(ctx, parentTaskNotify, parentTaskId, body, 0)
|
|
}
|
|
}
|
|
|
|
func notifyRemoteTask(ctx context.Context, notifyUrl string, taskid string, body jsonutils.JSONObject, tried int) {
|
|
client := httputils.GetDefaultClient()
|
|
header := http.Header{}
|
|
if len(taskid) > 0 {
|
|
header.Set("X-Task-Id", taskid)
|
|
}
|
|
_, body, err := httputils.JSONRequest(client, ctx, "POST", notifyUrl, header, body, true)
|
|
if err != nil {
|
|
log.Errorf("notifyRemoteTask fail %s", err)
|
|
if tried > MAX_REMOTE_NOTIFY_TRIES {
|
|
log.Errorf("notifyRemoteTask max tried reached, give up...")
|
|
} else {
|
|
notifyRemoteTask(ctx, notifyUrl, taskid, body, tried+1)
|
|
}
|
|
return
|
|
}
|
|
log.Infof("Notify remote URL %s(%s) get acked: %s!", notifyUrl, taskid, body.String())
|
|
}
|
|
|
|
func (self *STask) NotifyParentTaskFailure(ctx context.Context, reason jsonutils.JSONObject) {
|
|
body := jsonutils.NewDict()
|
|
body.Add(jsonutils.NewString("error"), "__status__")
|
|
body.Add(jsonutils.NewString(self.TaskName), "__task_name__")
|
|
body.Add(reason, "__reason__")
|
|
self.NotifyParentTaskComplete(ctx, body, true)
|
|
}
|
|
|
|
func (self *STask) IsCurrentStageComplete() bool {
|
|
totalSubtasks := SubTaskManager.GetTotalSubtasks(self.Id, self.Stage, "")
|
|
initSubtasks := SubTaskManager.GetInitSubtasks(self.Id, self.Stage)
|
|
if len(totalSubtasks) > 0 && len(initSubtasks) == 0 {
|
|
return true
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
|
|
func (self *STask) GetPendingUsage(quota quotas.IQuota, index int) error {
|
|
key := pendingUsageKey(index)
|
|
if self.Params.Contains(key) {
|
|
quotaJson, err := self.Params.Get(key)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "task.Params.Get %s", key)
|
|
}
|
|
err = quotaJson.Unmarshal(quota)
|
|
if err != nil {
|
|
return errors.Wrap(err, "quotaJson.Unmarshal")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func pendingUsageKey(index int) string {
|
|
key := PENDING_USAGE_KEY
|
|
if index > 0 {
|
|
key += "." + strconv.FormatInt(int64(index), 10)
|
|
}
|
|
return key
|
|
}
|
|
|
|
func (self *STask) SetPendingUsage(quota quotas.IQuota, index int) error {
|
|
_, err := db.Update(self, func() error {
|
|
key := pendingUsageKey(index)
|
|
params := self.Params.CopyExcludes(key)
|
|
params.Add(jsonutils.Marshal(quota), key)
|
|
self.Params = params
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
log.Errorf("set_pending_usage fail %s", err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (self *STask) ClearPendingUsage(index int) error {
|
|
_, err := db.Update(self, func() error {
|
|
key := pendingUsageKey(index)
|
|
params := self.Params.CopyExcludes(key)
|
|
self.Params = params
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
log.Errorf("clear_pending_usage fail %s", err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (self *STask) GetParams() *jsonutils.JSONDict {
|
|
return self.Params
|
|
}
|
|
|
|
func (self *STask) GetUserCred() mcclient.TokenCredential {
|
|
return self.UserCred
|
|
}
|
|
|
|
func (self *STask) GetTaskId() string {
|
|
return self.GetId()
|
|
}
|
|
|
|
func (self *STask) GetObject() db.IStandaloneModel {
|
|
return self.taskObject
|
|
}
|
|
|
|
func (self *STask) GetObjects() []db.IStandaloneModel {
|
|
return self.taskObjects
|
|
}
|
|
|
|
func (task *STask) GetTaskRequestHeader() http.Header {
|
|
userCred := task.GetUserCred()
|
|
if !userCred.IsValid() {
|
|
userCred = auth.AdminCredential()
|
|
}
|
|
header := mcclient.GetTokenHeaders(userCred)
|
|
header.Set(mcclient.TASK_ID, task.GetTaskId())
|
|
if len(serviceUrl) > 0 {
|
|
notifyUrl := fmt.Sprintf("%s/tasks/%s", serviceUrl, task.GetTaskId())
|
|
header.Set(mcclient.TASK_NOTIFY_URL, notifyUrl)
|
|
}
|
|
return header
|
|
}
|
|
|
|
var serviceUrl string
|
|
|
|
func SetServiceUrl(url string) {
|
|
serviceUrl = url
|
|
}
|
|
|
|
func (task *STask) GetStartTime() time.Time {
|
|
return task.CreatedAt
|
|
}
|
|
|
|
func (manager *STaskManager) QueryTasksOfObject(obj db.IStandaloneModel, since time.Time, isOpen *bool) *sqlchemy.SQuery {
|
|
subq1 := manager.Query()
|
|
{
|
|
subq1 = subq1.Equals("obj_id", obj.GetId())
|
|
subq1 = subq1.Equals("obj_name", obj.Keyword())
|
|
if !since.IsZero() {
|
|
subq1 = subq1.GE("created_at", since)
|
|
}
|
|
if isOpen != nil {
|
|
if *isOpen {
|
|
subq1 = subq1.Filter(sqlchemy.NOT(
|
|
sqlchemy.In(subq1.Field("stage"), []string{"complete", "failed"}),
|
|
))
|
|
} else if !*isOpen {
|
|
subq1 = subq1.In("stage", []string{"complete", "failed"})
|
|
}
|
|
}
|
|
}
|
|
|
|
subq2 := manager.Query()
|
|
{
|
|
taskObjs := TaskObjectManager.TableSpec().Instance()
|
|
subq2 = subq2.Join(taskObjs, sqlchemy.AND(
|
|
sqlchemy.Equals(taskObjs.Field("task_id"), subq2.Field("id")),
|
|
sqlchemy.Equals(taskObjs.Field("obj_id"), obj.GetId()),
|
|
))
|
|
subq2 = subq2.Filter(sqlchemy.Equals(subq2.Field("obj_id"), MULTI_OBJECTS_ID))
|
|
subq2 = subq2.Filter(sqlchemy.Equals(subq2.Field("obj_name"), obj.Keyword()))
|
|
if !since.IsZero() {
|
|
subq2 = subq2.Filter(sqlchemy.GE(subq2.Field("created_at"), since))
|
|
}
|
|
if isOpen != nil {
|
|
if *isOpen {
|
|
subq2 = subq2.Filter(sqlchemy.NOT(
|
|
sqlchemy.In(subq2.Field("stage"), []string{"complete", "failed"}),
|
|
))
|
|
} else if !*isOpen {
|
|
subq2 = subq2.In("stage", []string{"complete", "failed"})
|
|
}
|
|
}
|
|
}
|
|
|
|
// subq1 and subq2 do not overlap for the fact that they have
|
|
// different conditions on tasks_tbl.obj_id field
|
|
return sqlchemy.Union(subq1, subq2).Query().Desc("created_at")
|
|
}
|
|
|
|
func (manager *STaskManager) IsInTask(obj db.IStandaloneModel) bool {
|
|
tasks, err := manager.FetchIncompleteTasksOfObject(obj)
|
|
if err == nil && len(tasks) == 0 {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (manager *STaskManager) FetchIncompleteTasksOfObject(obj db.IStandaloneModel) ([]STask, error) {
|
|
isOpen := true
|
|
return manager.FetchTasksOfObjectLatest(obj, 1*time.Hour, &isOpen)
|
|
}
|
|
|
|
func (manager *STaskManager) FetchTasksOfObjectLatest(obj db.IStandaloneModel, interval time.Duration, isOpen *bool) ([]STask, error) {
|
|
since := timeutils.UtcNow().Add(-1 * interval)
|
|
return manager.FetchTasksOfObject(obj, since, isOpen)
|
|
}
|
|
|
|
func (manager *STaskManager) FetchTasksOfObject(obj db.IStandaloneModel, since time.Time, isOpen *bool) ([]STask, error) {
|
|
q := manager.QueryTasksOfObject(obj, since, isOpen)
|
|
|
|
tasks := make([]STask, 0)
|
|
err := db.FetchModelObjects(manager, q, &tasks)
|
|
if err != nil && err != sql.ErrNoRows {
|
|
return nil, err
|
|
}
|
|
return tasks, nil
|
|
}
|