mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-08 06:31:00 +08:00
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
176 lines
4.7 KiB
Go
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)
|
|
}
|