mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-21 14:12:57 -06:00
d60365af02
The previous mechanism for testing state threw away the mutation made on the state by calling State() twice - this commit corrects the test to match the comment. In addition, we replace the custom copying logic with the copystructure library to simplify the code.
153 lines
3.7 KiB
Go
153 lines
3.7 KiB
Go
package state
|
|
|
|
import (
|
|
"bytes"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
)
|
|
|
|
// TestState is a helper for testing state implementations. It is expected
|
|
// that the given implementation is pre-loaded with the TestStateInitial
|
|
// state.
|
|
func TestState(t *testing.T, s interface{}) {
|
|
reader, ok := s.(StateReader)
|
|
if !ok {
|
|
t.Fatalf("must at least be a StateReader")
|
|
}
|
|
|
|
// If it implements refresh, refresh
|
|
if rs, ok := s.(StateRefresher); ok {
|
|
if err := rs.RefreshState(); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
}
|
|
|
|
// current will track our current state
|
|
current := TestStateInitial()
|
|
|
|
// Check that the initial state is correct
|
|
if state := reader.State(); !current.Equal(state) {
|
|
t.Fatalf("not initial: %#v\n\n%#v", state, current)
|
|
}
|
|
|
|
// Write a new state and verify that we have it
|
|
if ws, ok := s.(StateWriter); ok {
|
|
current.Modules = append(current.Modules, &terraform.ModuleState{
|
|
Path: []string{"root"},
|
|
Outputs: map[string]*terraform.OutputState{
|
|
"bar": &terraform.OutputState{
|
|
Type: "string",
|
|
Sensitive: false,
|
|
Value: "baz",
|
|
},
|
|
},
|
|
})
|
|
|
|
if err := ws.WriteState(current); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
|
|
if actual := reader.State(); !actual.Equal(current) {
|
|
t.Fatalf("bad: %#v\n\n%#v", actual, current)
|
|
}
|
|
}
|
|
|
|
// Test persistence
|
|
if ps, ok := s.(StatePersister); ok {
|
|
if err := ps.PersistState(); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
|
|
// Refresh if we got it
|
|
if rs, ok := s.(StateRefresher); ok {
|
|
if err := rs.RefreshState(); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
}
|
|
|
|
// Just set the serials the same... Then compare.
|
|
actual := reader.State()
|
|
if !actual.Equal(current) {
|
|
t.Fatalf("bad: %#v\n\n%#v", actual, current)
|
|
}
|
|
}
|
|
|
|
// If we can write and persist then verify that the serial
|
|
// is only implemented on change.
|
|
writer, writeOk := s.(StateWriter)
|
|
persister, persistOk := s.(StatePersister)
|
|
if writeOk && persistOk {
|
|
// Same serial
|
|
serial := current.Serial
|
|
if err := writer.WriteState(current); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
if err := persister.PersistState(); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
|
|
if reader.State().Serial != serial {
|
|
t.Fatalf("bad: expected %d, got %d", serial, reader.State().Serial)
|
|
}
|
|
|
|
// Change the serial
|
|
currentCopy := *current
|
|
current = ¤tCopy
|
|
current.Modules = []*terraform.ModuleState{
|
|
&terraform.ModuleState{
|
|
Path: []string{"root", "somewhere"},
|
|
Outputs: map[string]*terraform.OutputState{
|
|
"serialCheck": &terraform.OutputState{
|
|
Type: "string",
|
|
Sensitive: false,
|
|
Value: "true",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
if err := writer.WriteState(current); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
if err := persister.PersistState(); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
|
|
if reader.State().Serial <= serial {
|
|
t.Fatalf("bad: expected %d, got %d", serial, reader.State().Serial)
|
|
}
|
|
|
|
// Check that State() returns a copy by modifying the copy and comparing
|
|
// to the current state.
|
|
stateCopy := reader.State()
|
|
stateCopy.Serial++
|
|
if reflect.DeepEqual(stateCopy, current) {
|
|
t.Fatal("State() should return a copy")
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestStateInitial is the initial state that a State should have
|
|
// for TestState.
|
|
func TestStateInitial() *terraform.State {
|
|
initial := &terraform.State{
|
|
Modules: []*terraform.ModuleState{
|
|
&terraform.ModuleState{
|
|
Path: []string{"root", "child"},
|
|
Outputs: map[string]*terraform.OutputState{
|
|
"foo": &terraform.OutputState{
|
|
Type: "string",
|
|
Sensitive: false,
|
|
Value: "bar",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
var scratch bytes.Buffer
|
|
terraform.WriteState(initial, &scratch)
|
|
return initial
|
|
}
|