mirror of
https://github.com/opentofu/opentofu.git
synced 2025-02-25 18:45:20 -06:00
This is part of a general effort to move all of Terraform's non-library package surface under internal in order to reinforce that these are for internal use within Terraform only. If you were previously importing packages under this prefix into an external codebase, you could pin to an earlier release tag as an interim solution until you've make a plan to achieve the same functionality some other way.
168 lines
4.0 KiB
Go
168 lines
4.0 KiB
Go
package azure
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/terraform/internal/backend"
|
|
"github.com/hashicorp/terraform/internal/states"
|
|
"github.com/hashicorp/terraform/internal/states/remote"
|
|
"github.com/hashicorp/terraform/internal/states/statemgr"
|
|
"github.com/tombuildsstuff/giovanni/storage/2018-11-09/blob/blobs"
|
|
"github.com/tombuildsstuff/giovanni/storage/2018-11-09/blob/containers"
|
|
)
|
|
|
|
const (
|
|
// This will be used as directory name, the odd looking colon is simply to
|
|
// reduce the chance of name conflicts with existing objects.
|
|
keyEnvPrefix = "env:"
|
|
)
|
|
|
|
func (b *Backend) Workspaces() ([]string, error) {
|
|
prefix := b.keyName + keyEnvPrefix
|
|
params := containers.ListBlobsInput{
|
|
Prefix: &prefix,
|
|
}
|
|
|
|
ctx := context.TODO()
|
|
client, err := b.armClient.getContainersClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
resp, err := client.ListBlobs(ctx, b.armClient.storageAccountName, b.containerName, params)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
envs := map[string]struct{}{}
|
|
for _, obj := range resp.Blobs.Blobs {
|
|
key := obj.Name
|
|
if strings.HasPrefix(key, prefix) {
|
|
name := strings.TrimPrefix(key, prefix)
|
|
// we store the state in a key, not a directory
|
|
if strings.Contains(name, "/") {
|
|
continue
|
|
}
|
|
|
|
envs[name] = struct{}{}
|
|
}
|
|
}
|
|
|
|
result := []string{backend.DefaultStateName}
|
|
for name := range envs {
|
|
result = append(result, name)
|
|
}
|
|
sort.Strings(result[1:])
|
|
return result, nil
|
|
}
|
|
|
|
func (b *Backend) DeleteWorkspace(name string) error {
|
|
if name == backend.DefaultStateName || name == "" {
|
|
return fmt.Errorf("can't delete default state")
|
|
}
|
|
|
|
ctx := context.TODO()
|
|
client, err := b.armClient.getBlobClient(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if resp, err := client.Delete(ctx, b.armClient.storageAccountName, b.containerName, b.path(name), blobs.DeleteInput{}); err != nil {
|
|
if resp.Response.StatusCode != 404 {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *Backend) StateMgr(name string) (statemgr.Full, error) {
|
|
ctx := context.TODO()
|
|
blobClient, err := b.armClient.getBlobClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
client := &RemoteClient{
|
|
giovanniBlobClient: *blobClient,
|
|
containerName: b.containerName,
|
|
keyName: b.path(name),
|
|
accountName: b.accountName,
|
|
snapshot: b.snapshot,
|
|
}
|
|
|
|
stateMgr := &remote.State{Client: client}
|
|
|
|
// Grab the value
|
|
if err := stateMgr.RefreshState(); err != nil {
|
|
return nil, err
|
|
}
|
|
//if this isn't the default state name, we need to create the object so
|
|
//it's listed by States.
|
|
if v := stateMgr.State(); v == nil {
|
|
// take a lock on this state while we write it
|
|
lockInfo := statemgr.NewLockInfo()
|
|
lockInfo.Operation = "init"
|
|
lockId, err := client.Lock(lockInfo)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to lock azure state: %s", err)
|
|
}
|
|
|
|
// Local helper function so we can call it multiple places
|
|
lockUnlock := func(parent error) error {
|
|
if err := stateMgr.Unlock(lockId); err != nil {
|
|
return fmt.Errorf(strings.TrimSpace(errStateUnlock), lockId, err)
|
|
}
|
|
return parent
|
|
}
|
|
|
|
// Grab the value
|
|
if err := stateMgr.RefreshState(); err != nil {
|
|
err = lockUnlock(err)
|
|
return nil, err
|
|
}
|
|
//if this isn't the default state name, we need to create the object so
|
|
//it's listed by States.
|
|
if v := stateMgr.State(); v == nil {
|
|
// If we have no state, we have to create an empty state
|
|
if err := stateMgr.WriteState(states.NewState()); err != nil {
|
|
err = lockUnlock(err)
|
|
return nil, err
|
|
}
|
|
if err := stateMgr.PersistState(); err != nil {
|
|
err = lockUnlock(err)
|
|
return nil, err
|
|
}
|
|
|
|
// Unlock, the state should now be initialized
|
|
if err := lockUnlock(nil); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
return stateMgr, nil
|
|
}
|
|
|
|
func (b *Backend) client() *RemoteClient {
|
|
return &RemoteClient{}
|
|
}
|
|
|
|
func (b *Backend) path(name string) string {
|
|
if name == backend.DefaultStateName {
|
|
return b.keyName
|
|
}
|
|
|
|
return b.keyName + keyEnvPrefix + name
|
|
}
|
|
|
|
const errStateUnlock = `
|
|
Error unlocking Azure state. Lock ID: %s
|
|
|
|
Error: %s
|
|
|
|
You may have to force-unlock this state in order to use it again.
|
|
`
|