|
|
@@ -19,8 +19,12 @@ package login
|
|
|
import (
|
|
|
"context"
|
|
|
"fmt"
|
|
|
+
|
|
|
"github.com/Azure/azure-sdk-for-go/services/storage/mgmt/2019-06-01/storage"
|
|
|
|
|
|
+ "github.com/docker/compose-cli/api/volumes"
|
|
|
+ "github.com/docker/compose-cli/errdefs"
|
|
|
+
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
|
"github.com/docker/compose-cli/context/store"
|
|
|
@@ -28,8 +32,7 @@ import (
|
|
|
|
|
|
// StorageAccountHelper helper for Azure Storage Account
|
|
|
type StorageAccountHelper struct {
|
|
|
- LoginService AzureLoginService
|
|
|
- AciContext store.AciContext
|
|
|
+ AciContext store.AciContext
|
|
|
}
|
|
|
|
|
|
// GetAzureStorageAccountKey retrieves the storage account ket from the current azure login
|
|
|
@@ -50,61 +53,99 @@ func (helper StorageAccountHelper) GetAzureStorageAccountKey(ctx context.Context
|
|
|
return *key.Value, nil
|
|
|
}
|
|
|
|
|
|
-func (helper StorageAccountHelper) ListFileShare(ctx context.Context) ([]string, error) {
|
|
|
+// ListFileShare list file shares in all visible storage accounts
|
|
|
+func (helper StorageAccountHelper) ListFileShare(ctx context.Context) ([]volumes.Volume, error) {
|
|
|
aciContext := helper.AciContext
|
|
|
accountClient, err := NewStorageAccountsClient(aciContext.SubscriptionID)
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
result, err := accountClient.ListByResourceGroup(ctx, aciContext.ResourceGroup)
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
accounts := result.Value
|
|
|
fileShareClient, err := NewFileShareClient(aciContext.SubscriptionID)
|
|
|
- fileShares := []string{}
|
|
|
+ if err != nil {
|
|
|
+ return nil, err
|
|
|
+ }
|
|
|
+ fileShares := []volumes.Volume{}
|
|
|
for _, account := range *accounts {
|
|
|
fileSharePage, err := fileShareClient.List(ctx, aciContext.ResourceGroup, *account.Name, "", "", "")
|
|
|
if err != nil {
|
|
|
return nil, err
|
|
|
}
|
|
|
- for ; fileSharePage.NotDone() ; fileSharePage.NextWithContext(ctx) {
|
|
|
+
|
|
|
+ for fileSharePage.NotDone() {
|
|
|
values := fileSharePage.Values()
|
|
|
for _, fileShare := range values {
|
|
|
- fileShares = append(fileShares, *fileShare.Name)
|
|
|
+ fileShares = append(fileShares, toVolume(account, *fileShare.Name))
|
|
|
+ }
|
|
|
+ if err := fileSharePage.NextWithContext(ctx); err != nil {
|
|
|
+ return nil, err
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return fileShares, nil
|
|
|
}
|
|
|
|
|
|
-func (helper StorageAccountHelper) CreateFileShare(ctx context.Context, accountName string, fileShareName string) (storage.FileShare, error) {
|
|
|
+func toVolume(account storage.Account, fileShareName string) volumes.Volume {
|
|
|
+ return volumes.Volume{
|
|
|
+ ID: fmt.Sprintf("%s@%s", *account.Name, fileShareName),
|
|
|
+ Name: fileShareName,
|
|
|
+ Description: fmt.Sprintf("Fileshare %s in %s storage account", fileShareName, *account.Name),
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+// CreateFileShare create a new fileshare
|
|
|
+func (helper StorageAccountHelper) CreateFileShare(ctx context.Context, accountName string, fileShareName string) (volumes.Volume, error) {
|
|
|
aciContext := helper.AciContext
|
|
|
accountClient, err := NewStorageAccountsClient(aciContext.SubscriptionID)
|
|
|
if err != nil {
|
|
|
- return storage.FileShare{}, err
|
|
|
+ return volumes.Volume{}, err
|
|
|
}
|
|
|
account, err := accountClient.GetProperties(ctx, aciContext.ResourceGroup, accountName, "")
|
|
|
if err != nil {
|
|
|
- //TODO check err not found
|
|
|
- parameters := storage.AccountCreateParameters{
|
|
|
- Location: &aciContext.Location,
|
|
|
- Sku:&storage.Sku{
|
|
|
- Name: storage.StandardLRS,
|
|
|
- Tier: storage.Standard,
|
|
|
- },
|
|
|
+ if account.StatusCode != 404 {
|
|
|
+ return volumes.Volume{}, err
|
|
|
}
|
|
|
+ //TODO confirm storage account creation
|
|
|
+ parameters := defaultStorageAccountParams(aciContext)
|
|
|
// TODO progress account creation
|
|
|
future, err := accountClient.Create(ctx, aciContext.ResourceGroup, accountName, parameters)
|
|
|
if err != nil {
|
|
|
- return storage.FileShare{}, err
|
|
|
+ return volumes.Volume{}, err
|
|
|
}
|
|
|
account, err = future.Result(accountClient)
|
|
|
+ if err != nil {
|
|
|
+ return volumes.Volume{}, err
|
|
|
+ }
|
|
|
}
|
|
|
fileShareClient, err := NewFileShareClient(aciContext.SubscriptionID)
|
|
|
- fileShare, err := fileShareClient.Get(ctx, aciContext.ResourceGroup, *account.Name, fileShareName, "")
|
|
|
if err != nil {
|
|
|
- // TODO check err not found
|
|
|
- fileShare, err = fileShareClient.Create(ctx, aciContext.ResourceGroup, *account.Name, fileShareName, storage.FileShare{})
|
|
|
+ return volumes.Volume{}, err
|
|
|
}
|
|
|
|
|
|
- return fileShare, nil
|
|
|
+ fileShare, err := fileShareClient.Get(ctx, aciContext.ResourceGroup, *account.Name, fileShareName, "")
|
|
|
+ if err == nil {
|
|
|
+ return volumes.Volume{}, errors.Wrapf(errdefs.ErrAlreadyExists, "Azure fileshare %q already exists", fileShareName)
|
|
|
+ }
|
|
|
+ if fileShare.StatusCode != 404 {
|
|
|
+ return volumes.Volume{}, err
|
|
|
+ }
|
|
|
+ fileShare, err = fileShareClient.Create(ctx, aciContext.ResourceGroup, *account.Name, fileShareName, storage.FileShare{})
|
|
|
+ if err != nil {
|
|
|
+ return volumes.Volume{}, err
|
|
|
+ }
|
|
|
+ return toVolume(account, *fileShare.Name), nil
|
|
|
}
|
|
|
|
|
|
+func defaultStorageAccountParams(aciContext store.AciContext) storage.AccountCreateParameters {
|
|
|
+ return storage.AccountCreateParameters{
|
|
|
+ Location: &aciContext.Location,
|
|
|
+ Sku: &storage.Sku{
|
|
|
+ Name: storage.StandardLRS,
|
|
|
+ Tier: storage.Standard,
|
|
|
+ },
|
|
|
+ }
|
|
|
+}
|