mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-06-01 05:22:24 +08:00
201 lines
5.0 KiB
Go
201 lines
5.0 KiB
Go
package openstack
|
||
|
||
import (
|
||
"fmt"
|
||
|
||
"yunion.io/x/jsonutils"
|
||
"yunion.io/x/log"
|
||
"yunion.io/x/onecloud/pkg/cloudcommon/version"
|
||
"yunion.io/x/onecloud/pkg/cloudprovider"
|
||
"yunion.io/x/onecloud/pkg/compute/models"
|
||
)
|
||
|
||
type ZoneState struct {
|
||
Available bool
|
||
}
|
||
|
||
type SZone struct {
|
||
region *SRegion
|
||
|
||
iwires []cloudprovider.ICloudWire
|
||
istorages []cloudprovider.ICloudStorage
|
||
|
||
ZoneName string
|
||
ZoneState ZoneState
|
||
}
|
||
|
||
func (zone *SZone) GetMetadata() *jsonutils.JSONDict {
|
||
return nil
|
||
}
|
||
|
||
func (zone *SZone) GetId() string {
|
||
return zone.ZoneName
|
||
}
|
||
|
||
func (zone *SZone) GetName() string {
|
||
return fmt.Sprintf("%s %s", CLOUD_PROVIDER_OPENSTACK, zone.ZoneName)
|
||
}
|
||
|
||
func (zone *SZone) GetGlobalId() string {
|
||
return fmt.Sprintf("%s/%s", zone.region.GetGlobalId(), zone.ZoneName)
|
||
}
|
||
|
||
func (zone *SZone) IsEmulated() bool {
|
||
return false
|
||
}
|
||
|
||
func (zone *SZone) GetStatus() string {
|
||
if zone.ZoneState.Available {
|
||
return models.ZONE_ENABLE
|
||
}
|
||
return models.ZONE_SOLDOUT
|
||
}
|
||
|
||
func (zone *SZone) Refresh() error {
|
||
// do nothing
|
||
return nil
|
||
}
|
||
|
||
func (zone *SZone) GetIRegion() cloudprovider.ICloudRegion {
|
||
return zone.region
|
||
}
|
||
|
||
func (zone *SZone) GetIWires() ([]cloudprovider.ICloudWire, error) {
|
||
return zone.iwires, nil
|
||
}
|
||
|
||
func (zone *SZone) getStorageByCategory(category string) (*SStorage, error) {
|
||
storages, err := zone.GetIStorages()
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
for i := 0; i < len(storages); i++ {
|
||
storage := storages[i].(*SStorage)
|
||
if storage.Name == category {
|
||
return storage, nil
|
||
}
|
||
}
|
||
return nil, fmt.Errorf("No such storage %s", category)
|
||
}
|
||
|
||
func (zone *SZone) addWire(wire *SWire) {
|
||
if zone.iwires == nil {
|
||
zone.iwires = []cloudprovider.ICloudWire{}
|
||
}
|
||
zone.iwires = append(zone.iwires, wire)
|
||
}
|
||
|
||
func (zone *SZone) fetchStorages() error {
|
||
zone.istorages = []cloudprovider.ICloudStorage{}
|
||
|
||
for _, service := range []string{"volumev3", "volumev2", "volume"} {
|
||
_, resp, err := zone.region.Get(service, "/types", "", nil)
|
||
if err == nil {
|
||
storages := []SStorage{}
|
||
if err := resp.Unmarshal(&storages, "volume_types"); err != nil {
|
||
return err
|
||
}
|
||
for i := 0; i < len(storages); i++ {
|
||
storages[i].zone = zone
|
||
zone.istorages = append(zone.istorages, &storages[i])
|
||
}
|
||
return nil
|
||
}
|
||
log.Debugf("failed to get volume types by service %s error: %v, try another", service, err)
|
||
}
|
||
return fmt.Errorf("failed to find storage types by cinder service")
|
||
}
|
||
|
||
func (zone *SZone) GetIStorages() ([]cloudprovider.ICloudStorage, error) {
|
||
if zone.istorages == nil {
|
||
zone.fetchStorages()
|
||
}
|
||
return zone.istorages, nil
|
||
}
|
||
|
||
func (zone *SZone) GetIStorageById(id string) (cloudprovider.ICloudStorage, error) {
|
||
if zone.istorages == nil {
|
||
zone.fetchStorages()
|
||
}
|
||
for i := 0; i < len(zone.istorages); i++ {
|
||
if zone.istorages[i].GetGlobalId() == id {
|
||
return zone.istorages[i], nil
|
||
}
|
||
}
|
||
return nil, cloudprovider.ErrNotFound
|
||
}
|
||
|
||
func (zone *SZone) GetIHosts() ([]cloudprovider.ICloudHost, error) {
|
||
// 2.28 Hypervisor CPU字段是字符串,会解析失败
|
||
_, maxVersion, err := zone.region.GetVersion("compute")
|
||
if err == nil && version.GE(maxVersion, "2.28") {
|
||
return zone.GetIHostsV3()
|
||
}
|
||
return zone.GetIHostsV2()
|
||
}
|
||
|
||
func (zone *SZone) GetIHostsV2() ([]cloudprovider.ICloudHost, error) {
|
||
_, resp, err := zone.region.Get("compute", "/os-hosts", "", nil)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
ihosts := []cloudprovider.ICloudHost{}
|
||
hosts := []SHostV2{}
|
||
if err := resp.Unmarshal(&hosts, "hosts"); err != nil {
|
||
return nil, err
|
||
}
|
||
for i := 0; i < len(hosts); i++ {
|
||
if hosts[i].Zone == zone.ZoneName {
|
||
hosts[i].zone = zone
|
||
ihosts = append(ihosts, &hosts[i])
|
||
}
|
||
}
|
||
return ihosts, nil
|
||
}
|
||
|
||
func (zone *SZone) GetIHostsV3() ([]cloudprovider.ICloudHost, error) {
|
||
_, maxVersion, err := zone.region.GetVersion("compute")
|
||
_, resp, err := zone.region.Get("compute", "/os-hypervisors/detail", maxVersion, nil)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
ihosts := []cloudprovider.ICloudHost{}
|
||
hosts := []SHostV3{}
|
||
if err := resp.Unmarshal(&hosts, "hypervisors"); err != nil {
|
||
return nil, err
|
||
}
|
||
for i := 0; i < len(hosts); i++ {
|
||
hosts[i].zone = zone
|
||
ihosts = append(ihosts, &hosts[i])
|
||
}
|
||
return ihosts, nil
|
||
}
|
||
|
||
func (zone *SZone) GetIHostById(id string) (cloudprovider.ICloudHost, error) {
|
||
_, maxVersion, err := zone.region.GetVersion("compute")
|
||
if err == nil && version.GE(maxVersion, "2.43") {
|
||
return zone.GetIHostByIdV3(id)
|
||
}
|
||
return zone.GetIHostByIdV2(id)
|
||
}
|
||
|
||
func (zone *SZone) GetIHostByIdV3(id string) (cloudprovider.ICloudHost, error) {
|
||
_, maxVersion, err := zone.region.GetVersion("compute")
|
||
_, resp, err := zone.region.Get("compute", "/os-hypervisors/"+id, maxVersion, nil)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
host := &SHostV3{zone: zone}
|
||
return host, resp.Unmarshal(&host, "hypervisor")
|
||
}
|
||
|
||
func (zone *SZone) GetIHostByIdV2(id string) (*SHostV2, error) {
|
||
_, maxVersion, err := zone.region.GetVersion("compute")
|
||
_, resp, err := zone.region.Get("compute", "/os-hosts/"+id, maxVersion, nil)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
host := &SHostV2{zone: zone, HostName: id, Resource: []map[string]SResource{}}
|
||
return host, resp.Unmarshal(&(host.Resource), "host")
|
||
}
|