package gcloud import ( "errors" "fmt" "path" "sort" "strings" "cloud.google.com/go/storage" "github.com/hashicorp/terraform/backend" "github.com/hashicorp/terraform/state" "github.com/hashicorp/terraform/state/remote" "github.com/hashicorp/terraform/terraform" "google.golang.org/api/iterator" ) // States returns a list of names for the states found on GCS. The default // state is always returned as the first element in the slice. func (b *Backend) States() ([]string, error) { states := []string{backend.DefaultStateName} bucket := b.storageClient.Bucket(b.bucketName) objs := bucket.Objects(b.storageContext, &storage.Query{ Delimiter: "/", Prefix: b.stateDir, }) for { attrs, err := objs.Next() if err == iterator.Done { break } if err != nil { return nil, fmt.Errorf("querying Cloud Storage failed: %v", err) } name := path.Base(attrs.Name) if !strings.HasSuffix(name, ".tfstate") { continue } st := strings.TrimSuffix(name, ".tfstate") if st != backend.DefaultStateName { states = append(states, st) } } sort.Strings(states[1:]) return states, nil } // DeleteState deletes the named state. The "default" state cannot be deleted. func (b *Backend) DeleteState(name string) error { if name == backend.DefaultStateName { return fmt.Errorf("cowardly refusing to delete the %q state", name) } client, err := b.remoteClient(name) if err != nil { return err } return client.Delete() } // get a remote client configured for this state func (b *Backend) remoteClient(name string) (*RemoteClient, error) { if name == "" { return nil, errors.New("Missing state name") } client := &RemoteClient{ storageContext: b.storageContext, storageClient: b.storageClient, bucketName: b.bucketName, stateFilePath: b.stateFileName(name), lockFilePath: b.lockFileName(name), } return client, nil } func (b *Backend) State(name string) (state.State, error) { client, err := b.remoteClient(name) if err != nil { return nil, fmt.Errorf("Failed to create Google Storage client: %v", err) } stateMgr := &remote.State{Client: client} lockInfo := state.NewLockInfo() lockInfo.Operation = "init" lockId, err := stateMgr.Lock(lockInfo) if err != nil { return nil, fmt.Errorf("Failed to lock state in Google Cloud Storage: %v", 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, client.lockFileURL(), err) } return parent } // Grab the value if err := stateMgr.RefreshState(); err != nil { err = lockUnlock(err) return nil, err } // If we have no state, we have to create an empty state if v := stateMgr.State(); v == nil { if err := stateMgr.WriteState(terraform.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) stateFileName(stateName string) string { if b.stateDir == "" { return fmt.Sprintf("%v.tfstate", stateName) } else { return fmt.Sprintf("%v/%v.tfstate", b.stateDir, stateName) } } func (b *Backend) lockFileName(stateName string) string { if b.stateDir == "" { return fmt.Sprintf("%v.tflock", stateName) } else { return fmt.Sprintf("%v/%v.tflock", b.stateDir, stateName) } } const errStateUnlock = ` Error unlocking Google Cloud Storage state. Lock ID: %v Lock file URL: %v Error: %v You may have to force-unlock this state in order to use it again. The GCloud backend acquires a lock during initialization to ensure the initial state file is created. `