Files
cloudpods/pkg/multicloud/openstack/zone.go
2021-05-28 20:18:10 +08:00

211 lines
5.1 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"
"strings"
"time"
"yunion.io/x/jsonutils"
"yunion.io/x/pkg/errors"
api "yunion.io/x/onecloud/pkg/apis/compute"
"yunion.io/x/onecloud/pkg/cloudprovider"
"yunion.io/x/onecloud/pkg/multicloud"
)
type ZoneState struct {
Available bool
}
type HostState struct {
Available bool
Active bool
UpdatedAt time.Time
}
type SZone struct {
multicloud.SResourceBase
multicloud.OpenStackTags
region *SRegion
ZoneName string
ZoneState ZoneState
Hosts map[string]jsonutils.JSONObject
hosts []SHypervisor
}
func (zone *SZone) GetId() string {
return zone.ZoneName
}
func (zone *SZone) GetName() string {
return zone.ZoneName
}
func (zone *SZone) GetI18n() cloudprovider.SModelI18nTable {
table := cloudprovider.SModelI18nTable{}
table["name"] = cloudprovider.NewSModelI18nEntry(zone.GetName()).CN(zone.GetName())
return table
}
func (zone *SZone) GetGlobalId() string {
return fmt.Sprintf("%s/%s/%s", CLOUD_PROVIDER_OPENSTACK, zone.region.Name, zone.ZoneName)
}
func (zone *SZone) IsEmulated() bool {
return false
}
func (zone *SZone) GetStatus() string {
if zone.ZoneState.Available {
return api.ZONE_ENABLE
}
return api.ZONE_DISABLE
}
func (zone *SZone) Refresh() error {
// do nothing
return nil
}
func (zone *SZone) GetIRegion() cloudprovider.ICloudRegion {
return zone.region
}
func (zone *SZone) getStorageByCategory(category, host string) (*SStorage, error) {
storages, err := zone.region.GetStorageTypes()
if err != nil {
return nil, errors.Wrap(err, "GetStorageTypes")
}
for i := range storages {
if storages[i].Name == category || storages[i].ExtraSpecs.VolumeBackendName == category {
storages[i].zone = zone
return &storages[i], nil
}
}
for i := range storages {
if strings.HasSuffix(host, "#"+storages[i].Name) || strings.HasSuffix(host, "#"+storages[i].ExtraSpecs.VolumeBackendName) {
storages[i].zone = zone
return &storages[i], nil
}
}
return nil, fmt.Errorf("No such storage [%s]", category)
}
func (zone *SZone) GetIStorages() ([]cloudprovider.ICloudStorage, error) {
storages, err := zone.region.GetStorageTypes()
if err != nil && errors.Cause(err) != ErrNoEndpoint {
return nil, errors.Wrap(err, "GetStorageTypes")
}
istorages := []cloudprovider.ICloudStorage{}
for i := range storages {
storages[i].zone = zone
istorages = append(istorages, &storages[i])
}
err = zone.fetchHosts()
if err != nil {
return nil, errors.Wrap(err, "fetchHosts")
}
for i := range zone.hosts {
nova := &SNovaStorage{host: &zone.hosts[i], zone: zone}
istorages = append(istorages, nova)
}
return istorages, nil
}
func (zone *SZone) GetIStorageById(id string) (cloudprovider.ICloudStorage, error) {
istorages, err := zone.GetIStorages()
if err != nil {
return nil, errors.Wrap(err, "GetIStorages")
}
for i := 0; i < len(istorages); i++ {
if istorages[i].GetGlobalId() == id {
return istorages[i], nil
}
}
return nil, cloudprovider.ErrNotFound
}
func (zone *SZone) fetchHosts() error {
if len(zone.hosts) > 0 {
return nil
}
zone.hosts = []SHypervisor{}
hypervisors, err := zone.region.GetHypervisors()
if err != nil {
return errors.Wrap(err, "GetHypervisors")
}
for i := range hypervisors {
hypervisor := strings.ToLower(hypervisors[i].HypervisorType)
// 过滤vmware的机器
if strings.Index(hypervisor, "vmware") != -1 {
continue
}
_, ok1 := zone.Hosts[hypervisors[i].HypervisorHostname]
_, ok2 := zone.Hosts[hypervisors[i].Service.Host]
if !ok1 && !ok2 {
continue
}
zone.hosts = append(zone.hosts, hypervisors[i])
}
return nil
}
func (zone *SZone) GetIHosts() ([]cloudprovider.ICloudHost, error) {
err := zone.fetchHosts()
if err != nil {
return nil, errors.Wrap(err, "fetchHosts")
}
ihosts := []cloudprovider.ICloudHost{}
for i := range zone.hosts {
zone.hosts[i].zone = zone
ihosts = append(ihosts, &zone.hosts[i])
}
return ihosts, nil
}
func (zone *SZone) GetIHostById(id string) (cloudprovider.ICloudHost, error) {
ihosts, err := zone.GetIHosts()
if err != nil {
return nil, errors.Wrap(err, "GetIHosts")
}
for i := range ihosts {
if ihosts[i].GetGlobalId() == id {
return ihosts[i], nil
}
}
return nil, cloudprovider.ErrNotFound
}
func (region *SRegion) getZones() ([]SZone, error) {
zones := []SZone{}
resp, err := region.ecsList("os-availability-zone/detail", nil)
if err != nil {
return nil, errors.Wrap(err, "ecsList.os-availability-zone")
}
err = resp.Unmarshal(&zones, "availabilityZoneInfo")
if err != nil {
return nil, errors.Wrap(err, "resp.Unmarshal")
}
return zones, nil
}