mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-23 21:11:02 +08:00
258 lines
6.3 KiB
Go
258 lines
6.3 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 openstack
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"strings"
|
|
"time"
|
|
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/errors"
|
|
"yunion.io/x/pkg/utils"
|
|
|
|
api "yunion.io/x/onecloud/pkg/apis/compute"
|
|
"yunion.io/x/onecloud/pkg/cloudprovider"
|
|
"yunion.io/x/onecloud/pkg/multicloud"
|
|
)
|
|
|
|
const (
|
|
DEFAULT_STORAGE_TYPE = "scheduler"
|
|
)
|
|
|
|
type SExtraSpecs struct {
|
|
VolumeBackendName string
|
|
}
|
|
|
|
type SStorage struct {
|
|
multicloud.SStorageBase
|
|
zone *SZone
|
|
Name string
|
|
ExtraSpecs SExtraSpecs
|
|
ID string
|
|
}
|
|
|
|
func (storage *SStorage) GetId() string {
|
|
return storage.ID
|
|
}
|
|
|
|
func (storage *SStorage) GetName() string {
|
|
return storage.Name
|
|
}
|
|
|
|
func (storage *SStorage) GetGlobalId() string {
|
|
return fmt.Sprintf("%s-%s", storage.zone.GetGlobalId(), storage.ID)
|
|
}
|
|
|
|
func (storage *SStorage) IsEmulated() bool {
|
|
return false
|
|
}
|
|
|
|
func (storage *SStorage) GetIZone() cloudprovider.ICloudZone {
|
|
return storage.zone
|
|
}
|
|
|
|
func (storage *SStorage) GetIDisks() ([]cloudprovider.ICloudDisk, error) {
|
|
disks, err := storage.zone.region.GetDisks()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
idisks := []cloudprovider.ICloudDisk{}
|
|
for i := 0; i < len(disks); i++ {
|
|
if disks[i].AvailabilityZone == storage.zone.ZoneName && (disks[i].VolumeType == storage.Name || strings.HasSuffix(disks[i].Host, "#"+storage.ExtraSpecs.VolumeBackendName)) {
|
|
disks[i].storage = storage
|
|
idisks = append(idisks, &disks[i])
|
|
}
|
|
}
|
|
return idisks, nil
|
|
}
|
|
|
|
func (storage *SStorage) GetStorageType() string {
|
|
if len(storage.ExtraSpecs.VolumeBackendName) == 0 {
|
|
return DEFAULT_STORAGE_TYPE
|
|
}
|
|
return storage.ExtraSpecs.VolumeBackendName
|
|
}
|
|
|
|
func (storage *SStorage) GetMediumType() string {
|
|
if strings.Contains(storage.Name, "SSD") {
|
|
return api.DISK_TYPE_SSD
|
|
}
|
|
return api.DISK_TYPE_ROTATE
|
|
}
|
|
|
|
func (storage *SStorage) GetCapacityMB() int64 {
|
|
return 0 // unlimited
|
|
}
|
|
|
|
func (storage *SStorage) GetCapacityUsedMB() int64 {
|
|
return 0
|
|
}
|
|
|
|
func (storage *SStorage) GetStorageConf() jsonutils.JSONObject {
|
|
conf := jsonutils.NewDict()
|
|
return conf
|
|
}
|
|
|
|
func (storage *SStorage) GetStatus() string {
|
|
ok, err := storage.zone.region.IsStorageAvailable(storage.GetStorageType())
|
|
if err != nil || !ok {
|
|
return api.STORAGE_OFFLINE
|
|
}
|
|
return api.STORAGE_ONLINE
|
|
}
|
|
|
|
func (storage *SStorage) Refresh() error {
|
|
// do nothing
|
|
return nil
|
|
}
|
|
|
|
func (storage *SStorage) GetEnabled() bool {
|
|
return true
|
|
}
|
|
|
|
func (storage *SStorage) GetIStoragecache() cloudprovider.ICloudStoragecache {
|
|
return storage.zone.region.getStoragecache()
|
|
}
|
|
|
|
func (storage *SStorage) CreateIDisk(conf *cloudprovider.DiskCreateConfig) (cloudprovider.ICloudDisk, error) {
|
|
disk, err := storage.zone.region.CreateDisk("", storage.Name, conf.Name, conf.SizeGb, conf.Desc, conf.ProjectId)
|
|
if err != nil {
|
|
log.Errorf("createDisk fail %v", err)
|
|
return nil, err
|
|
}
|
|
disk.storage = storage
|
|
return disk, cloudprovider.WaitStatus(disk, api.DISK_READY, time.Second*5, time.Minute*5)
|
|
}
|
|
|
|
func (storage *SStorage) GetIDiskById(idStr string) (cloudprovider.ICloudDisk, error) {
|
|
disk, err := storage.zone.region.GetDisk(idStr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if disk.AvailabilityZone != storage.zone.ZoneName {
|
|
return nil, errors.Wrapf(cloudprovider.ErrNotFound, "disk %s not in zone %s", disk.Name, storage.zone.ZoneName)
|
|
}
|
|
disk.storage = storage
|
|
return disk, nil
|
|
}
|
|
|
|
func (storage *SStorage) GetMountPoint() string {
|
|
return ""
|
|
}
|
|
|
|
func (storage *SStorage) IsSysDiskStore() bool {
|
|
return true
|
|
}
|
|
|
|
func (region *SRegion) GetStorageTypes() ([]SStorage, error) {
|
|
resource := "/types"
|
|
storages := []SStorage{}
|
|
query := url.Values{}
|
|
for {
|
|
resp, err := region.bsList(resource, query)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "bsReqest")
|
|
}
|
|
part := struct {
|
|
VolumeTypes []SStorage
|
|
VolumeTypeLinks SNextLinks
|
|
}{}
|
|
err = resp.Unmarshal(&part)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "resp.Unmarshal")
|
|
}
|
|
storages = append(storages, part.VolumeTypes...)
|
|
marker := part.VolumeTypeLinks.GetNextMark()
|
|
if len(marker) == 0 {
|
|
break
|
|
}
|
|
query.Set("marker", marker)
|
|
}
|
|
return storages, nil
|
|
}
|
|
|
|
type SCinderService struct {
|
|
ActiveBackendId string
|
|
// cinder-volume
|
|
Binary string
|
|
DisabledReason string
|
|
Frozen string
|
|
Host string
|
|
ReplicationStatus string
|
|
State string
|
|
Status string
|
|
UpdatedAt time.Time
|
|
Zone string
|
|
}
|
|
|
|
func (region *SRegion) GetCinderServices() ([]SCinderService, error) {
|
|
resp, err := region.bsList("/os-services", nil)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "bsList")
|
|
}
|
|
services := []SCinderService{}
|
|
err = resp.Unmarshal(&services, "services")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "resp.Unmarshal")
|
|
}
|
|
return services, nil
|
|
}
|
|
|
|
func (region *SRegion) IsStorageAvailable(storageType string) (bool, error) {
|
|
if utils.IsInStringArray(storageType, []string{DEFAULT_STORAGE_TYPE, api.STORAGE_OPENSTACK_NOVA}) {
|
|
return true, nil
|
|
}
|
|
services, err := region.GetCinderServices()
|
|
if err != nil {
|
|
return false, errors.Wrap(err, "GetCinderServices")
|
|
}
|
|
for _, service := range services {
|
|
if service.Binary == "cinder-volume" && strings.Contains(service.Host, "@") {
|
|
hostInfo := strings.Split(service.Host, "@")
|
|
if hostInfo[len(hostInfo)-1] == storageType {
|
|
if service.State == "up" && service.Status == "enabled" {
|
|
return true, nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
log.Errorf("storage %s offline", storageType)
|
|
return false, nil
|
|
}
|
|
|
|
type SCapabilities struct {
|
|
}
|
|
|
|
type SPool struct {
|
|
Name string
|
|
Capabilities SCapabilities
|
|
}
|
|
|
|
func (region *SRegion) GetSchedulerStatsPool() ([]SPool, error) {
|
|
resp, err := region.bsList("/scheduler-stats/get_pools", nil)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "bsList")
|
|
}
|
|
pools := []SPool{}
|
|
err = resp.Unmarshal(&pools, "pools")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "resp.Unmarshal")
|
|
}
|
|
return pools, nil
|
|
}
|