// 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 guestdrivers import ( "context" "database/sql" "fmt" "yunion.io/x/cloudmux/pkg/cloudprovider" "yunion.io/x/log" "yunion.io/x/pkg/errors" "yunion.io/x/pkg/util/billing" "yunion.io/x/pkg/util/cloudinit" "yunion.io/x/pkg/util/rbacscope" "yunion.io/x/pkg/utils" "yunion.io/x/sqlchemy" api "yunion.io/x/onecloud/pkg/apis/compute" "yunion.io/x/onecloud/pkg/cloudcommon/db" "yunion.io/x/onecloud/pkg/cloudcommon/db/quotas" "yunion.io/x/onecloud/pkg/cloudcommon/db/taskman" "yunion.io/x/onecloud/pkg/compute/models" "yunion.io/x/onecloud/pkg/mcclient" ) type SSangForGuestDriver struct { SManagedVirtualizedGuestDriver } func init() { driver := SSangForGuestDriver{} models.RegisterGuestDriver(&driver) } func (self *SSangForGuestDriver) GetHypervisor() string { return api.HYPERVISOR_SANGFOR } func (self *SSangForGuestDriver) GetProvider() string { return api.CLOUD_PROVIDER_SANGFOR } func (self *SSangForGuestDriver) DoScheduleSKUFilter() bool { return false } func (self *SSangForGuestDriver) DoScheduleStorageFilter() bool { return false } func (self *SSangForGuestDriver) GetComputeQuotaKeys(scope rbacscope.TRbacScope, ownerId mcclient.IIdentityProvider, brand string) models.SComputeResourceKeys { keys := models.SComputeResourceKeys{} keys.SBaseProjectQuotaKeys = quotas.OwnerIdProjectQuotaKeys(scope, ownerId) keys.CloudEnv = api.CLOUD_ENV_PRIVATE_CLOUD keys.Provider = api.CLOUD_PROVIDER_SANGFOR keys.Brand = api.CLOUD_PROVIDER_SANGFOR keys.Hypervisor = api.HYPERVISOR_SANGFOR return keys } func (self *SSangForGuestDriver) GetDefaultSysDiskBackend() string { return api.STORAGE_LOCAL } func (self *SSangForGuestDriver) GetMinimalSysDiskSizeGb() int { return 40 } func (self *SSangForGuestDriver) GetStorageTypes() []string { storages, _ := models.StorageManager.GetStorageTypesByProvider(self.GetProvider()) return storages } func (self *SSangForGuestDriver) ChooseHostStorage(host *models.SHost, guest *models.SGuest, diskConfig *api.DiskConfig, storageIds []string) (*models.SStorage, error) { return chooseHostStorage(self, host, diskConfig.Backend, storageIds), nil } func (self *SSangForGuestDriver) GetDetachDiskStatus() ([]string, error) { return []string{api.VM_READY, api.VM_RUNNING}, nil } func (self *SSangForGuestDriver) GetAttachDiskStatus() ([]string, error) { return []string{api.VM_READY, api.VM_RUNNING}, nil } func (self *SSangForGuestDriver) GetRebuildRootStatus() ([]string, error) { return []string{api.VM_READY}, nil } func (self *SSangForGuestDriver) GetChangeInstanceTypeStatus() ([]string, error) { return []string{api.VM_READY}, nil } func (self *SSangForGuestDriver) GetDeployStatus() ([]string, error) { return []string{api.VM_READY, api.VM_RUNNING}, nil } func (self *SSangForGuestDriver) ValidateResizeDisk(guest *models.SGuest, disk *models.SDisk, storage *models.SStorage) error { if !utils.IsInStringArray(guest.Status, []string{api.VM_RUNNING, api.VM_READY}) { return fmt.Errorf("Cannot resize disk when guest in status %s", guest.Status) } return nil } func (self *SSangForGuestDriver) ValidateCreateData(ctx context.Context, userCred mcclient.TokenCredential, input *api.ServerCreateInput) (*api.ServerCreateInput, error) { if len(input.UserData) > 0 { _, err := cloudinit.ParseUserData(input.UserData) if err != nil { return nil, err } } if len(input.Cdrom) > 0 { image, err := models.CachedimageManager.GetCachedimageById(ctx, userCred, input.Cdrom, false) if err != nil { return nil, err } if len(image.ExternalId) > 0 { hosts, err := image.GetHosts() if err != nil { return nil, err } if len(input.PreferHost) == 0 && len(hosts) == 1 { input.PreferHost = hosts[0].Id } } } return input, nil } func (self *SSangForGuestDriver) GetGuestInitialStateAfterCreate() string { return api.VM_READY } func (self *SSangForGuestDriver) DoGuestCreateDisksTask(ctx context.Context, guest *models.SGuest, task taskman.ITask) error { subtask, err := taskman.TaskManager.NewTask(ctx, "SangForGuestCreateDiskTask", guest, task.GetUserCred(), task.GetParams(), task.GetTaskId(), "", nil) if err != nil { return errors.Wrapf(err, "NewTask") } return subtask.ScheduleRun(nil) } func (self *SSangForGuestDriver) GetGuestInitialStateAfterRebuild() string { return api.VM_READY } func (self *SSangForGuestDriver) GetInstanceCapability() cloudprovider.SInstanceCapability { return cloudprovider.SInstanceCapability{ Hypervisor: self.GetHypervisor(), Provider: self.GetProvider(), DefaultAccount: cloudprovider.SDefaultAccount{ Linux: cloudprovider.SOsDefaultAccount{ DefaultAccount: api.VM_DEFAULT_LINUX_LOGIN_USER, }, Windows: cloudprovider.SOsDefaultAccount{ DefaultAccount: api.VM_DEFAULT_WINDOWS_LOGIN_USER, }, }, } } func (self *SSangForGuestDriver) RemoteDeployGuestSyncHost(ctx context.Context, userCred mcclient.TokenCredential, guest *models.SGuest, host *models.SHost, iVM cloudprovider.ICloudVM) (cloudprovider.ICloudHost, error) { if hostId := iVM.GetIHostId(); len(hostId) > 0 { nh, err := db.FetchByExternalIdAndManagerId(models.HostManager, hostId, func(q *sqlchemy.SQuery) *sqlchemy.SQuery { return q.Equals("manager_id", host.ManagerId) }) if err != nil { log.Debugf("failed to found new hostId(%s) for ivm %s(%s) error: %v", hostId, guest.Name, guest.Id, err) if errors.Cause(err) != sql.ErrNoRows { return nil, errors.Wrap(err, "FetchByExternalIdAndManagerId") } // HYPERVISOR_SANGFOR VM被部署到一台全新的宿主机 zone, err := host.GetZone() if err != nil { log.Warningf("host %s GetZone: %s", host.GetId(), err) } else { _host, err := models.HostManager.NewFromCloudHost(ctx, userCred, iVM.GetIHost(), host.GetCloudprovider(), zone) if err != nil { log.Warningf("NewFromCloudHost %s: %s", iVM.GetIHostId(), err) } else { host = _host } } } else { host = nh.(*models.SHost) } } if host.GetId() != guest.HostId { guest.OnScheduleToHost(ctx, userCred, host.GetId()) } return host.GetIHost(ctx) } func (self *SSangForGuestDriver) IsSupportedBillingCycle(bc billing.SBillingCycle) bool { return false } func (self *SSangForGuestDriver) IsNeedInjectPasswordByCloudInit() bool { return false } func (self *SSangForGuestDriver) IsSupportSetAutoRenew() bool { return false } func (self *SSangForGuestDriver) RequestSyncSecgroupsOnHost(ctx context.Context, guest *models.SGuest, host *models.SHost, task taskman.ITask) error { return nil // do nothing, not support securitygroup } func (self *SSangForGuestDriver) GetMaxSecurityGroupCount() int { //暂不支持绑定安全组 return 0 } func (self *SSangForGuestDriver) RequestGuestHotAddIso(ctx context.Context, guest *models.SGuest, path string, boot bool, task taskman.ITask) error { task.ScheduleRun(nil) return nil } func (self *SSangForGuestDriver) IsSupportCdrom(guest *models.SGuest) (bool, error) { return true, nil }