2015-02-21 13:52:55 -06:00
|
|
|
package remote
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2017-05-25 10:01:25 -05:00
|
|
|
"sync"
|
2015-02-21 13:52:55 -06:00
|
|
|
|
2017-02-15 16:20:59 -06:00
|
|
|
"github.com/hashicorp/terraform/state"
|
2015-02-21 13:52:55 -06:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
)
|
|
|
|
|
|
|
|
// State implements the State interfaces in the state package to handle
|
|
|
|
// reading and writing the remote state. This State on its own does no
|
|
|
|
// local caching so every persist will go to the remote storage and local
|
|
|
|
// writes will go to memory.
|
|
|
|
type State struct {
|
2017-05-25 10:01:25 -05:00
|
|
|
mu sync.Mutex
|
|
|
|
|
2015-02-21 13:52:55 -06:00
|
|
|
Client Client
|
|
|
|
|
2015-02-23 23:30:59 -06:00
|
|
|
state, readState *terraform.State
|
2015-02-21 13:52:55 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// StateReader impl.
|
|
|
|
func (s *State) State() *terraform.State {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-02-23 23:36:35 -06:00
|
|
|
return s.state.DeepCopy()
|
2015-02-21 13:52:55 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// StateWriter impl.
|
|
|
|
func (s *State) WriteState(state *terraform.State) error {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-02-21 13:52:55 -06:00
|
|
|
s.state = state
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// StateRefresher impl.
|
|
|
|
func (s *State) RefreshState() error {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-02-21 13:52:55 -06:00
|
|
|
payload, err := s.Client.Get()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-07-06 09:48:52 -05:00
|
|
|
// no remote state is OK
|
2016-07-01 17:37:38 -05:00
|
|
|
if payload == nil {
|
2016-07-06 09:48:52 -05:00
|
|
|
return nil
|
2016-07-01 17:37:38 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
state, err := terraform.ReadState(bytes.NewReader(payload.Data))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2015-02-21 13:52:55 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
s.state = state
|
2015-02-23 23:30:59 -06:00
|
|
|
s.readState = state
|
2015-02-21 13:52:55 -06:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// StatePersister impl.
|
|
|
|
func (s *State) PersistState() error {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-02-23 23:30:59 -06:00
|
|
|
s.state.IncrementSerialMaybe(s.readState)
|
|
|
|
|
2015-02-21 13:52:55 -06:00
|
|
|
var buf bytes.Buffer
|
|
|
|
if err := terraform.WriteState(s.state, &buf); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.Client.Put(buf.Bytes())
|
|
|
|
}
|
2017-01-09 17:18:31 -06:00
|
|
|
|
|
|
|
// Lock calls the Client's Lock method if it's implemented.
|
2017-02-15 16:20:59 -06:00
|
|
|
func (s *State) Lock(info *state.LockInfo) (string, error) {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2017-02-15 16:20:59 -06:00
|
|
|
if c, ok := s.Client.(ClientLocker); ok {
|
|
|
|
return c.Lock(info)
|
2017-01-09 17:18:31 -06:00
|
|
|
}
|
2017-02-15 16:20:59 -06:00
|
|
|
return "", nil
|
2017-01-09 17:18:31 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unlock calls the Client's Unlock method if it's implemented.
|
2017-02-15 16:20:59 -06:00
|
|
|
func (s *State) Unlock(id string) error {
|
2017-05-25 10:01:25 -05:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2017-02-15 16:20:59 -06:00
|
|
|
if c, ok := s.Client.(ClientLocker); ok {
|
|
|
|
return c.Unlock(id)
|
2017-01-09 17:18:31 -06:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|