2015-02-21 20:52:55 +01:00
|
|
|
package state
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
)
|
|
|
|
|
|
|
|
// LocalState manages a state storage that is local to the filesystem.
|
|
|
|
type LocalState struct {
|
2015-02-22 01:11:47 +01:00
|
|
|
// Path is the path to read the state from. PathOut is the path to
|
|
|
|
// write the state to. If PathOut is not specified, Path will be used.
|
|
|
|
// If PathOut already exists, it will be overwritten.
|
|
|
|
Path string
|
|
|
|
PathOut string
|
2015-02-21 20:52:55 +01:00
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
state *terraform.State
|
|
|
|
written bool
|
2015-02-21 20:52:55 +01:00
|
|
|
}
|
|
|
|
|
2015-02-22 03:00:08 +01:00
|
|
|
// SetState will force a specific state in-memory for this local state.
|
|
|
|
func (s *LocalState) SetState(state *terraform.State) {
|
|
|
|
s.state = state
|
|
|
|
}
|
|
|
|
|
2015-02-21 20:52:55 +01:00
|
|
|
// StateReader impl.
|
|
|
|
func (s *LocalState) State() *terraform.State {
|
|
|
|
return s.state
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteState for LocalState always persists the state as well.
|
|
|
|
//
|
|
|
|
// StateWriter impl.
|
|
|
|
func (s *LocalState) WriteState(state *terraform.State) error {
|
|
|
|
s.state = state
|
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
path := s.PathOut
|
|
|
|
if path == "" {
|
|
|
|
path = s.Path
|
|
|
|
}
|
|
|
|
|
2015-02-22 03:00:08 +01:00
|
|
|
// If we don't have any state, we actually delete the file if it exists
|
|
|
|
if state == nil {
|
|
|
|
err := os.Remove(path)
|
|
|
|
if err != nil && os.IsNotExist(err) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
f, err := os.Create(path)
|
2015-02-21 20:52:55 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
if err := terraform.WriteState(s.state, f); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
s.written = true
|
|
|
|
return nil
|
2015-02-21 20:52:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// PersistState for LocalState is a no-op since WriteState always persists.
|
|
|
|
//
|
|
|
|
// StatePersister impl.
|
|
|
|
func (s *LocalState) PersistState() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// StateRefresher impl.
|
|
|
|
func (s *LocalState) RefreshState() error {
|
2015-02-22 01:11:47 +01:00
|
|
|
// If we've never loaded before, read from Path, otherwise we
|
|
|
|
// read from PathOut.
|
|
|
|
path := s.Path
|
|
|
|
if s.written && s.PathOut != "" {
|
|
|
|
path = s.PathOut
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(path)
|
2015-02-21 20:52:55 +01:00
|
|
|
if err != nil {
|
2015-02-22 01:06:27 +01:00
|
|
|
// It is okay if the file doesn't exist, we treat that as a nil state
|
2015-02-22 01:11:47 +01:00
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
return err
|
2015-02-22 01:06:27 +01:00
|
|
|
}
|
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
f = nil
|
2015-02-21 20:52:55 +01:00
|
|
|
}
|
|
|
|
|
2015-02-22 01:11:47 +01:00
|
|
|
var state *terraform.State
|
|
|
|
if f != nil {
|
|
|
|
defer f.Close()
|
|
|
|
state, err = terraform.ReadState(f)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-02-21 20:52:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
s.state = state
|
|
|
|
return nil
|
|
|
|
}
|