Files
cloudpods/pkg/devtool/models/script_apply.go
rainzm fd458a6030 feat(devtool): makes the process of installing the agent more rigorous
1. Use the method returned by server sshable to execute ansible playbook
2. Use direct connection and cloud proxy to find the url that the remote machine
can send data back to local influxdb
2021-04-27 11:58:34 +08:00

176 lines
4.7 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 models
import (
"context"
"fmt"
"sync"
"yunion.io/x/jsonutils"
"yunion.io/x/pkg/errors"
"yunion.io/x/pkg/util/sets"
api "yunion.io/x/onecloud/pkg/apis/devtool"
"yunion.io/x/onecloud/pkg/cloudcommon/db"
"yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
"yunion.io/x/onecloud/pkg/mcclient"
)
type SScriptApply struct {
db.SStatusStandaloneResourceBase
ScriptId string `width:"36" nullable:"false" index:"true"`
GuestId string `width:"36" nullable:"false" index:"true"`
//
Args jsonutils.JSONObject
TryTimes int
ArgsGenerator string `width:"36" nullable:"false"`
}
type SScriptApplyManager struct {
db.SStatusStandaloneResourceBaseManager
Session *sScriptApplySession
}
var ScriptApplyManager *SScriptApplyManager
func init() {
ScriptApplyManager = &SScriptApplyManager{
SStatusStandaloneResourceBaseManager: db.NewStatusStandaloneResourceBaseManager(
SScriptApply{},
"scriptapply_tbl",
"scriptapply",
"scirptapplys",
),
Session: newScriptApplySession(),
}
ScriptApplyManager.SetVirtualObject(ScriptApplyManager)
}
func (sam *SScriptApplyManager) createScriptApply(ctx context.Context, scriptId, guestId string, args map[string]interface{}, argsGenerator string) (*SScriptApply, error) {
sa := &SScriptApply{
ScriptId: scriptId,
GuestId: guestId,
Args: jsonutils.Marshal(args),
ArgsGenerator: argsGenerator,
}
err := ScriptApplyManager.TableSpec().Insert(ctx, sa)
sa.SetModelManager(ScriptApplyManager, sa)
return sa, err
}
func (sa *SScriptApply) StartApply(ctx context.Context, userCred mcclient.TokenCredential) (err error) {
if ok := ScriptApplyManager.Session.CheckAndSet(sa.Id); !ok {
return fmt.Errorf("script %s is applying to server %s", sa.ScriptId, sa.GuestId)
}
defer func() {
if err != nil {
ScriptApplyManager.Session.Remove(sa.Id)
}
}()
// check try times
script, err := sa.Script()
if err != nil {
return err
}
if sa.TryTimes >= script.MaxTryTimes {
return fmt.Errorf("The times to try has exceeded the maximum times %d setted by the script", script.MaxTryTimes)
}
_, err = db.Update(sa, func() error {
sa.TryTimes += 1
sa.Status = api.SCRIPT_APPLY_STATUS_APPLYING
return nil
})
if err != nil {
return errors.Wrap(err, "unable to update scriptapply")
}
err = sa.startApplyScriptTask(ctx, userCred, "")
if err != nil {
f := false
_, err = ScriptApplyRecordManager.createRecordWithResult(ctx, sa.GetId(), &f, fmt.Sprintf("unabel to start ApplyScriptTask: %v", err))
if err != nil {
return errors.Wrap(err, "unable to record")
}
ScriptApplyManager.Session.Remove(sa.Id)
}
return nil
}
func (sa *SScriptApply) startApplyScriptTask(ctx context.Context, userCred mcclient.TokenCredential, parentTaskId string) error {
task, err := taskman.TaskManager.NewTask(ctx, "ApplyScriptTask", sa, userCred, nil, "", parentTaskId)
if err != nil {
return err
}
task.ScheduleRun(nil)
return nil
}
func (sa *SScriptApply) StopApply(userCred mcclient.TokenCredential, record *SScriptApplyRecord, success bool, failCode string, reason string) error {
var status string
if success {
status = api.SCRIPT_APPLY_STATUS_READY
if record != nil {
record.Succeed(reason)
}
} else {
status = api.SCRIPT_APPLY_RECORD_FAILED
if record != nil {
record.Fail(failCode, reason)
}
}
sa.SetStatus(userCred, status, "")
ScriptApplyManager.Session.Remove(sa.Id)
return nil
}
func (sa *SScriptApply) Script() (*SScript, error) {
obj, err := ScriptManager.FetchById(sa.ScriptId)
if err != nil {
return nil, errors.Wrapf(err, "unable to fetch Script %s", sa.Id)
}
s := obj.(*SScript)
s.SetModelManager(ScriptManager, s)
return s, nil
}
type sScriptApplySession struct {
mux *sync.Mutex
applyingOnes sets.String
}
func newScriptApplySession() *sScriptApplySession {
return &sScriptApplySession{
mux: &sync.Mutex{},
applyingOnes: sets.NewString(),
}
}
func (sas *sScriptApplySession) CheckAndSet(id string) bool {
sas.mux.Lock()
defer sas.mux.Unlock()
if sas.applyingOnes.Has(id) {
return false
}
sas.applyingOnes.Insert(id)
return true
}
func (sas *sScriptApplySession) Remove(id string) {
sas.mux.Lock()
defer sas.mux.Unlock()
sas.applyingOnes.Delete(id)
}