Files
cloudpods/pkg/multicloud/openstack/keypaire.go

108 lines
2.8 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"
"yunion.io/x/jsonutils"
"yunion.io/x/pkg/utils"
"yunion.io/x/onecloud/pkg/cloudprovider"
"yunion.io/x/onecloud/pkg/util/version"
)
type SKeypair struct {
Fingerprint string
Name string
Type string
PublicKey string
}
type SKeyPair struct {
Keypair SKeypair
}
func (region *SRegion) GetKeypairs() ([]SKeyPair, error) {
_, resp, err := region.List("compute", "/os-keypairs", "", nil)
if err != nil {
return nil, err
}
keypairs := []SKeyPair{}
return keypairs, resp.Unmarshal(&keypairs, "keypairs")
}
func (region *SRegion) CreateKeypair(name, publicKey, Type string) (*SKeyPair, error) {
if len(Type) > 0 && !utils.IsInStringArray(Type, []string{"ssh", "x509"}) {
return nil, fmt.Errorf("only support ssh or x509 type")
}
params := map[string]map[string]string{
"keypair": {
"name": name,
"public_key": publicKey,
},
}
_, maxVersion, _ := region.GetVersion("compute")
if len(Type) > 0 && version.GE(maxVersion, "2.2") {
params["keypair"]["type"] = Type
}
_, resp, err := region.Post("compute", "/os-keypairs", maxVersion, jsonutils.Marshal(params))
if err != nil {
return nil, err
}
keypair := &SKeyPair{}
return keypair, resp.Unmarshal(keypair)
}
func (region *SRegion) DeleteKeypair(name string) error {
_, err := region.Delete("compute", "/os-keypairs/"+name, "")
return err
}
func (region *SRegion) GetKeypair(name string) (*SKeyPair, error) {
_, resp, err := region.Get("compute", "/os-keypairs/"+name, "", nil)
if err != nil {
return nil, err
}
keypair := &SKeyPair{}
return keypair, resp.Unmarshal(keypair)
}
func (region *SRegion) syncKeypair(namePrefix, publicKey string) (string, error) {
keypairs, err := region.GetKeypairs()
if err != nil {
return "", err
}
for _, keypair := range keypairs {
if keypair.Keypair.PublicKey == publicKey {
return keypair.Keypair.Name, nil
}
}
for i := 0; i < 10; i++ {
name := fmt.Sprintf("%s-%d", namePrefix, i)
if _, err := region.GetKeypair(name); err != nil {
if err == cloudprovider.ErrNotFound {
keypair, err := region.CreateKeypair(name, publicKey, "ssh")
if err != nil {
return "", err
}
return keypair.Keypair.Name, nil
}
}
}
return "", fmt.Errorf("failed to find uniq name for keypair")
}