mirror of https://github.com/k3s-io/k3s
150 lines
5.2 KiB
Go
150 lines
5.2 KiB
Go
|
// +build !providerless
|
||
|
|
||
|
/*
|
||
|
Copyright 2016 The Kubernetes Authors.
|
||
|
|
||
|
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 azure
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/Azure/azure-sdk-for-go/services/storage/mgmt/2019-04-01/storage"
|
||
|
"github.com/Azure/go-autorest/autorest/to"
|
||
|
|
||
|
"k8s.io/klog"
|
||
|
)
|
||
|
|
||
|
type accountWithLocation struct {
|
||
|
Name, StorageType, Location string
|
||
|
}
|
||
|
|
||
|
// getStorageAccounts gets name, type, location of all storage accounts in a resource group which matches matchingAccountType, matchingLocation
|
||
|
func (az *Cloud) getStorageAccounts(matchingAccountType, matchingAccountKind, resourceGroup, matchingLocation string) ([]accountWithLocation, error) {
|
||
|
ctx, cancel := getContextWithCancel()
|
||
|
defer cancel()
|
||
|
result, err := az.StorageAccountClient.ListByResourceGroup(ctx, resourceGroup)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if result.Value == nil {
|
||
|
return nil, fmt.Errorf("unexpected error when listing storage accounts from resource group %s", resourceGroup)
|
||
|
}
|
||
|
|
||
|
accounts := []accountWithLocation{}
|
||
|
for _, acct := range *result.Value {
|
||
|
if acct.Name != nil && acct.Location != nil && acct.Sku != nil {
|
||
|
storageType := string((*acct.Sku).Name)
|
||
|
if matchingAccountType != "" && !strings.EqualFold(matchingAccountType, storageType) {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
if matchingAccountKind != "" && !strings.EqualFold(matchingAccountKind, string(acct.Kind)) {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
location := *acct.Location
|
||
|
if matchingLocation != "" && !strings.EqualFold(matchingLocation, location) {
|
||
|
continue
|
||
|
}
|
||
|
accounts = append(accounts, accountWithLocation{Name: *acct.Name, StorageType: storageType, Location: location})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return accounts, nil
|
||
|
}
|
||
|
|
||
|
// GetStorageAccesskey gets the storage account access key
|
||
|
func (az *Cloud) GetStorageAccesskey(account, resourceGroup string) (string, error) {
|
||
|
ctx, cancel := getContextWithCancel()
|
||
|
defer cancel()
|
||
|
|
||
|
result, err := az.StorageAccountClient.ListKeys(ctx, resourceGroup, account)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
if result.Keys == nil {
|
||
|
return "", fmt.Errorf("empty keys")
|
||
|
}
|
||
|
|
||
|
for _, k := range *result.Keys {
|
||
|
if k.Value != nil && *k.Value != "" {
|
||
|
v := *k.Value
|
||
|
if ind := strings.LastIndex(v, " "); ind >= 0 {
|
||
|
v = v[(ind + 1):]
|
||
|
}
|
||
|
return v, nil
|
||
|
}
|
||
|
}
|
||
|
return "", fmt.Errorf("no valid keys")
|
||
|
}
|
||
|
|
||
|
// EnsureStorageAccount search storage account, create one storage account(with genAccountNamePrefix) if not found, return accountName, accountKey
|
||
|
func (az *Cloud) EnsureStorageAccount(accountName, accountType, accountKind, resourceGroup, location, genAccountNamePrefix string) (string, string, error) {
|
||
|
if len(accountName) == 0 {
|
||
|
// find a storage account that matches accountType
|
||
|
accounts, err := az.getStorageAccounts(accountType, accountKind, resourceGroup, location)
|
||
|
if err != nil {
|
||
|
return "", "", fmt.Errorf("could not list storage accounts for account type %s: %v", accountType, err)
|
||
|
}
|
||
|
|
||
|
if len(accounts) > 0 {
|
||
|
accountName = accounts[0].Name
|
||
|
klog.V(4).Infof("found a matching account %s type %s location %s", accounts[0].Name, accounts[0].StorageType, accounts[0].Location)
|
||
|
}
|
||
|
|
||
|
if len(accountName) == 0 {
|
||
|
// not found a matching account, now create a new account in current resource group
|
||
|
accountName = generateStorageAccountName(genAccountNamePrefix)
|
||
|
if location == "" {
|
||
|
location = az.Location
|
||
|
}
|
||
|
if accountType == "" {
|
||
|
accountType = defaultStorageAccountType
|
||
|
}
|
||
|
|
||
|
// use StorageV2 by default per https://docs.microsoft.com/en-us/azure/storage/common/storage-account-options
|
||
|
kind := defaultStorageAccountKind
|
||
|
if accountKind != "" {
|
||
|
kind = storage.Kind(accountKind)
|
||
|
}
|
||
|
klog.V(2).Infof("azure - no matching account found, begin to create a new account %s in resource group %s, location: %s, accountType: %s, accountKind: %s",
|
||
|
accountName, resourceGroup, location, accountType, kind)
|
||
|
cp := storage.AccountCreateParameters{
|
||
|
Sku: &storage.Sku{Name: storage.SkuName(accountType)},
|
||
|
Kind: kind,
|
||
|
AccountPropertiesCreateParameters: &storage.AccountPropertiesCreateParameters{EnableHTTPSTrafficOnly: to.BoolPtr(true)},
|
||
|
Tags: map[string]*string{"created-by": to.StringPtr("azure")},
|
||
|
Location: &location}
|
||
|
|
||
|
ctx, cancel := getContextWithCancel()
|
||
|
defer cancel()
|
||
|
_, err := az.StorageAccountClient.Create(ctx, resourceGroup, accountName, cp)
|
||
|
if err != nil {
|
||
|
return "", "", fmt.Errorf(fmt.Sprintf("Failed to create storage account %s, error: %s", accountName, err))
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// find the access key with this account
|
||
|
accountKey, err := az.GetStorageAccesskey(accountName, resourceGroup)
|
||
|
if err != nil {
|
||
|
return "", "", fmt.Errorf("could not get storage key for storage account %s: %v", accountName, err)
|
||
|
}
|
||
|
|
||
|
return accountName, accountKey, nil
|
||
|
}
|