| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624 |
- // 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"
- "io"
- "net/url"
- "time"
- "yunion.io/x/jsonutils"
- "yunion.io/x/pkg/errors"
- "yunion.io/x/pkg/util/httputils"
- api "yunion.io/x/cloudmux/pkg/apis/compute"
- "yunion.io/x/cloudmux/pkg/cloudprovider"
- "yunion.io/x/cloudmux/pkg/multicloud"
- )
- type SRegion struct {
- multicloud.SRegion
- client *SOpenStackClient
- Name string
- vpcs []SVpc
- storageCache *SStoragecache
- routers []SRouter
- }
- func (region *SRegion) GetClient() *SOpenStackClient {
- return region.client
- }
- func (region *SRegion) GetId() string {
- return region.Name
- }
- func (region *SRegion) GetName() string {
- return fmt.Sprintf("%s-%s", region.client.cpcfg.Name, region.Name)
- }
- func (region *SRegion) GetI18n() cloudprovider.SModelI18nTable {
- table := cloudprovider.SModelI18nTable{}
- table["name"] = cloudprovider.NewSModelI18nEntry(region.GetName()).CN(region.GetName())
- return table
- }
- func (region *SRegion) GetGlobalId() string {
- return fmt.Sprintf("%s/%s/%s", CLOUD_PROVIDER_OPENSTACK, region.client.cpcfg.Id, region.Name)
- }
- func (region *SRegion) IsEmulated() bool {
- return false
- }
- func (region *SRegion) GetProvider() string {
- return CLOUD_PROVIDER_OPENSTACK
- }
- func (region *SRegion) GetCloudEnv() string {
- return ""
- }
- func (region *SRegion) GetGeographicInfo() cloudprovider.SGeographicInfo {
- return cloudprovider.SGeographicInfo{}
- }
- func (region *SRegion) GetStatus() string {
- return api.CLOUD_REGION_STATUS_INSERVER
- }
- func (region *SRegion) Refresh() error {
- // do nothing
- return nil
- }
- func (region *SRegion) GetMaxVersion(service string) (string, error) {
- return region.client.GetMaxVersion(region.Name, service)
- }
- func (region *SRegion) CreateIVpc(opts *cloudprovider.VpcCreateOptions) (cloudprovider.ICloudVpc, error) {
- vpc, err := region.CreateVpc(opts.NAME, opts.Desc)
- if err != nil {
- return nil, errors.Wrap(err, "CreateVp")
- }
- return vpc, nil
- }
- func (region *SRegion) GetIHostById(id string) (cloudprovider.ICloudHost, error) {
- izones, err := region.GetIZones()
- if err != nil {
- return nil, err
- }
- for i := 0; i < len(izones); i++ {
- ihost, err := izones[i].GetIHostById(id)
- if err == nil {
- return ihost, nil
- } else if errors.Cause(err) != cloudprovider.ErrNotFound {
- return nil, err
- }
- }
- return nil, cloudprovider.ErrNotFound
- }
- func (region *SRegion) GetIStorageById(id string) (cloudprovider.ICloudStorage, error) {
- izones, err := region.GetIZones()
- if err != nil {
- return nil, err
- }
- for i := 0; i < len(izones); i++ {
- istore, err := izones[i].GetIStorageById(id)
- if err == nil {
- return istore, nil
- } else if errors.Cause(err) != cloudprovider.ErrNotFound {
- return nil, err
- }
- }
- return nil, cloudprovider.ErrNotFound
- }
- func (region *SRegion) GetIHosts() ([]cloudprovider.ICloudHost, error) {
- iHosts := []cloudprovider.ICloudHost{}
- izones, err := region.GetIZones()
- if err != nil {
- return nil, err
- }
- for i := 0; i < len(izones); i++ {
- iZoneHost, err := izones[i].GetIHosts()
- if err != nil {
- return nil, err
- }
- iHosts = append(iHosts, iZoneHost...)
- }
- return iHosts, nil
- }
- func (region *SRegion) GetIStorages() ([]cloudprovider.ICloudStorage, error) {
- izones, err := region.GetIZones()
- if err != nil {
- return nil, err
- }
- iStorages := []cloudprovider.ICloudStorage{}
- for i := 0; i < len(izones); i++ {
- iZoneStores, err := izones[i].GetIStorages()
- if err != nil {
- return nil, err
- }
- iStorages = append(iStorages, iZoneStores...)
- }
- return iStorages, nil
- }
- func (region *SRegion) getStoragecache() *SStoragecache {
- if region.storageCache == nil {
- region.storageCache = &SStoragecache{region: region}
- }
- return region.storageCache
- }
- func (region *SRegion) GetIStoragecacheById(id string) (cloudprovider.ICloudStoragecache, error) {
- storageCache := region.getStoragecache()
- if storageCache.GetGlobalId() == id {
- return region.storageCache, nil
- }
- return nil, cloudprovider.ErrNotFound
- }
- func (region *SRegion) GetIStoragecaches() ([]cloudprovider.ICloudStoragecache, error) {
- storageCache := region.getStoragecache()
- return []cloudprovider.ICloudStoragecache{storageCache}, nil
- }
- func (region *SRegion) GetIVMById(id string) (cloudprovider.ICloudVM, error) {
- instance, err := region.GetInstance(id)
- if err != nil {
- return nil, errors.Wrapf(err, "GetInstance(%s)", id)
- }
- hosts, err := region.GetIHosts()
- if err != nil {
- return nil, err
- }
- for i := range hosts {
- host := hosts[i].(*SHypervisor)
- if instance.HypervisorHostname == host.HypervisorHostname {
- instance.host = host
- }
- }
- return instance, nil
- }
- func (region *SRegion) GetIDiskById(id string) (cloudprovider.ICloudDisk, error) {
- disk, err := region.GetDisk(id)
- if err != nil {
- _, err := region.GetInstance(id)
- if err == nil {
- return &SNovaDisk{region: region, instanceId: id}, nil
- }
- return nil, errors.Wrapf(err, "GetDisk(%s)", id)
- }
- return disk, nil
- }
- func (region *SRegion) GetIVpcById(id string) (cloudprovider.ICloudVpc, error) {
- vpc, err := region.GetVpc(id)
- if err != nil {
- return nil, errors.Wrapf(err, "GetVpc(%s)", id)
- }
- return vpc, nil
- }
- func (region *SRegion) GetIZoneById(id string) (cloudprovider.ICloudZone, error) {
- izones, err := region.GetIZones()
- if err != nil {
- return nil, errors.Wrap(err, "GetIZones")
- }
- for i := 0; i < len(izones); i++ {
- if izones[i].GetGlobalId() == id {
- return izones[i], nil
- }
- }
- return nil, cloudprovider.ErrNotFound
- }
- func (region *SRegion) GetZones() ([]SZone, error) {
- zones, err := region.getZones()
- if err != nil {
- return nil, errors.Wrap(err, "getZones")
- }
- ret := []SZone{}
- for i := 0; i < len(zones); i++ {
- if zones[i].ZoneName == "internal" {
- continue
- }
- zones[i].region = region
- ret = append(ret, zones[i])
- }
- return ret, nil
- }
- func (region *SRegion) fetchVpcs() error {
- if len(region.vpcs) > 0 {
- return nil
- }
- var err error
- region.vpcs, err = region.GetVpcs("")
- if err != nil {
- return errors.Wrap(err, "GetVpcs")
- }
- return nil
- }
- func (region *SRegion) ecsList(resource string, query url.Values) (jsonutils.JSONObject, error) {
- return region.client.ecsRequest(region.Name, httputils.GET, resource, query, nil)
- }
- func (region *SRegion) ecsGet(resource string) (jsonutils.JSONObject, error) {
- return region.client.ecsRequest(region.Name, httputils.GET, resource, nil, nil)
- }
- func (region *SRegion) ecsUpdate(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.ecsRequest(region.Name, httputils.PUT, resource, nil, params)
- }
- func (region *SRegion) ecsPost(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.ecsRequest(region.Name, httputils.POST, resource, nil, params)
- }
- func (region *SRegion) ecsDo(projectId, resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.ecsDo(projectId, region.Name, resource, params)
- }
- func (region *SRegion) ecsDelete(resource string) (jsonutils.JSONObject, error) {
- return region.client.ecsRequest(region.Name, httputils.DELETE, resource, nil, nil)
- }
- func (region *SRegion) ecsCreate(projectId, resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.ecsCreate(projectId, region.Name, resource, params)
- }
- func (region *SRegion) vpcList(resource string, query url.Values) (jsonutils.JSONObject, error) {
- return region.client.vpcRequest(region.Name, httputils.GET, resource, query, nil)
- }
- func (region *SRegion) vpcGet(resource string) (jsonutils.JSONObject, error) {
- return region.client.vpcRequest(region.Name, httputils.GET, resource, nil, nil)
- }
- func (region *SRegion) vpcUpdate(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.vpcRequest(region.Name, httputils.PUT, resource, nil, params)
- }
- func (region *SRegion) vpcPost(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.vpcRequest(region.Name, httputils.POST, resource, nil, params)
- }
- func (region *SRegion) vpcDelete(resource string) (jsonutils.JSONObject, error) {
- return region.client.vpcRequest(region.Name, httputils.DELETE, resource, nil, nil)
- }
- func (region *SRegion) imageList(resource string, query url.Values) (jsonutils.JSONObject, error) {
- return region.client.imageRequest(region.Name, httputils.GET, resource, query, nil)
- }
- func (region *SRegion) imageGet(resource string) (jsonutils.JSONObject, error) {
- return region.client.imageRequest(region.Name, httputils.GET, resource, nil, nil)
- }
- func (region *SRegion) imagePost(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.imageRequest(region.Name, httputils.POST, resource, nil, params)
- }
- func (region *SRegion) imageDelete(resource string) (jsonutils.JSONObject, error) {
- return region.client.imageRequest(region.Name, httputils.DELETE, resource, nil, nil)
- }
- func (region *SRegion) imageUpload(url string, size int64, body io.Reader, callback func(progress float32)) error {
- resp, err := region.client.imageUpload(region.Name, url, size, body, callback)
- _, _, err = httputils.ParseResponse("", resp, err, region.client.debug)
- return err
- }
- // Block Storage
- func (region *SRegion) bsList(resource string, query url.Values) (jsonutils.JSONObject, error) {
- return region.client.bsRequest(region.Name, httputils.GET, resource, query, nil)
- }
- func (region *SRegion) bsGet(resource string) (jsonutils.JSONObject, error) {
- return region.client.bsRequest(region.Name, httputils.GET, resource, nil, nil)
- }
- func (region *SRegion) bsUpdate(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.bsRequest(region.Name, httputils.PUT, resource, nil, params)
- }
- func (region *SRegion) bsPost(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.bsRequest(region.Name, httputils.POST, resource, nil, params)
- }
- func (region *SRegion) bsDelete(resource string) (jsonutils.JSONObject, error) {
- return region.client.bsRequest(region.Name, httputils.DELETE, resource, nil, nil)
- }
- func (region *SRegion) bsCreate(projectId, resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.bsCreate(projectId, region.Name, resource, params)
- }
- //loadbalancer
- func (region *SRegion) lbList(resource string, query url.Values) (jsonutils.JSONObject, error) {
- return region.client.lbRequest(region.Name, httputils.GET, resource, query, nil)
- }
- func (region *SRegion) lbGet(resource string) (jsonutils.JSONObject, error) {
- return region.client.lbRequest(region.Name, httputils.GET, resource, nil, nil)
- }
- func (region *SRegion) lbUpdate(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.lbRequest(region.Name, httputils.PUT, resource, nil, params)
- }
- func (region *SRegion) lbPost(resource string, params interface{}) (jsonutils.JSONObject, error) {
- return region.client.lbRequest(region.Name, httputils.POST, resource, nil, params)
- }
- func (region *SRegion) lbDelete(resource string) (jsonutils.JSONObject, error) {
- return region.client.lbRequest(region.Name, httputils.DELETE, resource, nil, nil)
- }
- func (region *SRegion) ProjectId() string {
- return region.client.tokenCredential.GetProjectId()
- }
- func (region *SRegion) GetIZones() ([]cloudprovider.ICloudZone, error) {
- zones, err := region.GetZones()
- if err != nil {
- return nil, errors.Wrapf(err, "GetZones")
- }
- ret := []cloudprovider.ICloudZone{}
- for i := range zones {
- ret = append(ret, &zones[i])
- }
- return ret, nil
- }
- func (region *SRegion) GetIVpcs() ([]cloudprovider.ICloudVpc, error) {
- err := region.fetchVpcs()
- if err != nil {
- return nil, errors.Wrap(err, "fetchVpcs")
- }
- ivpcs := []cloudprovider.ICloudVpc{}
- for i := range region.vpcs {
- region.vpcs[i].region = region
- ivpcs = append(ivpcs, ®ion.vpcs[i])
- }
- return ivpcs, nil
- }
- func (region *SRegion) GetIEips() ([]cloudprovider.ICloudEIP, error) {
- eips, err := region.GetEips("")
- if err != nil {
- return nil, err
- }
- ieips := []cloudprovider.ICloudEIP{}
- for i := 0; i < len(eips); i++ {
- eips[i].region = region
- ieips = append(ieips, &eips[i])
- }
- return ieips, nil
- }
- func (region *SRegion) CreateEIP(eip *cloudprovider.SEip) (cloudprovider.ICloudEIP, error) {
- network, err := region.GetNetwork(eip.NetworkExternalId)
- if err != nil {
- return nil, errors.Wrapf(err, "GetNetwork(%s)", eip.NetworkExternalId)
- }
- ieip, err := region.CreateEip(network.NetworkId, eip.NetworkExternalId, eip.Ip, eip.ProjectId)
- if err != nil {
- return nil, errors.Wrap(err, "CreateEip")
- }
- return ieip, nil
- }
- func (region *SRegion) GetIEipById(eipId string) (cloudprovider.ICloudEIP, error) {
- return region.GetEip(eipId)
- }
- func (region *SRegion) GetILoadBalancers() ([]cloudprovider.ICloudLoadbalancer, error) {
- loadbalancers := []cloudprovider.ICloudLoadbalancer{}
- sloadbalancers, err := region.GetLoadbalancers()
- if err != nil {
- return nil, errors.Wrap(err, "region.GetLoadbalancers()")
- }
- for i := 0; i < len(sloadbalancers); i++ {
- loadbalancers = append(loadbalancers, &sloadbalancers[i])
- }
- return loadbalancers, nil
- }
- func (region *SRegion) GetILoadBalancerById(loadbalancerId string) (cloudprovider.ICloudLoadbalancer, error) {
- sloadbalancer, err := region.GetLoadbalancerbyId(loadbalancerId)
- if err != nil {
- return nil, errors.Wrapf(err, "region.GetLoadbalancerbyId(%s)", loadbalancerId)
- }
- return sloadbalancer, nil
- }
- func (region *SRegion) GetILoadBalancerAclById(aclId string) (cloudprovider.ICloudLoadbalancerAcl, error) {
- return region.GetLoadbalancerAclDetail(aclId)
- }
- func (region *SRegion) GetILoadBalancerCertificateById(certId string) (cloudprovider.ICloudLoadbalancerCertificate, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) CreateILoadBalancerCertificate(cert *cloudprovider.SLoadbalancerCertificate) (cloudprovider.ICloudLoadbalancerCertificate, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) GetILoadBalancerAcls() ([]cloudprovider.ICloudLoadbalancerAcl, error) {
- iloadbalancerAcls := []cloudprovider.ICloudLoadbalancerAcl{}
- acls, err := region.GetLoadBalancerAcls()
- if err != nil {
- return nil, errors.Wrap(err, "region.GetLoadBalancerAcls")
- }
- for i := 0; i < len(acls); i++ {
- iloadbalancerAcls = append(iloadbalancerAcls, &acls[i])
- }
- return iloadbalancerAcls, nil
- }
- func (region *SRegion) GetILoadBalancerCertificates() ([]cloudprovider.ICloudLoadbalancerCertificate, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) CreateILoadBalancer(loadbalancer *cloudprovider.SLoadbalancerCreateOptions) (cloudprovider.ICloudLoadbalancer, error) {
- sloadbalancer, err := region.CreateLoadBalancer(loadbalancer)
- if err != nil {
- return nil, errors.Wrap(err, "region.CreateLoadBalancer")
- }
- return sloadbalancer, nil
- }
- func (region *SRegion) CreateILoadBalancerAcl(acl *cloudprovider.SLoadbalancerAccessControlList) (cloudprovider.ICloudLoadbalancerAcl, error) {
- sacl, err := region.CreateLoadBalancerAcl(acl)
- if err != nil {
- return nil, errors.Wrap(err, "region.CreateLoadBalancerAcl(acl)")
- }
- err = cloudprovider.WaitMultiStatus(sacl.listener, []string{api.LB_STATUS_ENABLED, api.LB_STATUS_UNKNOWN}, 10*time.Second, 8*time.Minute)
- if err != nil {
- return nil, errors.Wrap(err, "cloudprovider.WaitMultiStatus")
- }
- if sacl.listener.GetStatus() == api.LB_STATUS_UNKNOWN {
- return nil, errors.Wrap(fmt.Errorf("status error"), "check status")
- }
- return sacl, nil
- }
- func (region *SRegion) GetISkus() ([]cloudprovider.ICloudSku, error) {
- flavors, err := region.GetFlavors()
- if err != nil {
- return nil, err
- }
- iskus := make([]cloudprovider.ICloudSku, len(flavors))
- for i := 0; i < len(flavors); i++ {
- flavors[i].region = region
- iskus[i] = &flavors[i]
- }
- return iskus, nil
- }
- func (region *SRegion) GetIBuckets() ([]cloudprovider.ICloudBucket, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) CreateIBucket(name string, storageClassStr string, acl string) error {
- return cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) DeleteIBucket(name string) error {
- return cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) IBucketExist(name string) (bool, error) {
- return false, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) GetIBucketById(name string) (cloudprovider.ICloudBucket, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) GetIBucketByName(name string) (cloudprovider.ICloudBucket, error) {
- return nil, cloudprovider.ErrNotImplemented
- }
- func (region *SRegion) GetISecurityGroupById(secgroupId string) (cloudprovider.ICloudSecurityGroup, error) {
- return region.GetSecurityGroup(secgroupId)
- }
- func (region *SRegion) GetISecurityGroups() ([]cloudprovider.ICloudSecurityGroup, error) {
- err := region.client.fetchProjects()
- if err != nil {
- return nil, errors.Wrap(err, "fetchProjects")
- }
- iSecgroups := []cloudprovider.ICloudSecurityGroup{}
- for _, project := range region.client.projects {
- secgroups, err := region.GetSecurityGroups(project.Id, "")
- if err != nil {
- return nil, errors.Wrapf(err, "GetSecurityGroups(%s)", project.Id)
- }
- for i := 0; i < len(secgroups); i++ {
- secgroups[i].region = region
- iSecgroups = append(iSecgroups, &secgroups[i])
- }
- }
- return iSecgroups, nil
- }
- func (region *SRegion) CreateISecurityGroup(opts *cloudprovider.SecurityGroupCreateInput) (cloudprovider.ICloudSecurityGroup, error) {
- return region.CreateSecurityGroup(opts)
- }
- func (region *SRegion) GetCapabilities() []string {
- return region.client.GetCapabilities()
- }
- func (region *SRegion) GetRouters() ([]SRouter, error) {
- resource := "/v2.0/routers"
- resp, err := region.vpcList(resource, nil)
- if err != nil {
- return nil, errors.Wrap(err, "vpcList.routes")
- }
- routers := []SRouter{}
- err = resp.Unmarshal(&routers, "routers")
- if err != nil {
- return nil, errors.Wrap(err, "resp.Unmarshal")
- }
- for i := 0; i < len(routers); i++ {
- ports, err := region.GetPorts("", routers[i].Id)
- if err != nil {
- return nil, errors.Wrap(err, "vpc.region.GetPortsByDeviceId")
- }
- routers[i].ports = ports
- }
- return routers, nil
- }
- func (region *SRegion) fetchrouters() error {
- if len(region.routers) > 0 {
- return nil
- }
- routers, err := region.GetRouters()
- if err != nil {
- return errors.Wrap(err, "region.GetRouters()")
- }
- region.routers = routers
- return nil
- }
- func (region *SRegion) GetIVMs() ([]cloudprovider.ICloudVM, error) {
- vms, err := region.GetInstances("")
- if err != nil {
- return nil, errors.Wrapf(err, "GetInstances")
- }
- ret := []cloudprovider.ICloudVM{}
- for i := range vms {
- ret = append(ret, &vms[i])
- }
- return ret, nil
- }
|