Files
coder/provisioner/terraform/timings_test.go
2025-01-29 14:35:04 +00:00

137 lines
6.0 KiB
Go

//go:build linux || darwin
package terraform_test
import (
"context"
"os"
"path/filepath"
"testing"
"github.com/stretchr/testify/require"
"github.com/coder/coder/v2/coderd/database"
terraform_internal "github.com/coder/coder/v2/provisioner/terraform/internal"
"github.com/coder/coder/v2/provisionersdk/proto"
"github.com/coder/coder/v2/testutil"
)
// TestTimingsFromProvision uses a fake terraform binary which spits out expected log content.
// This log content is then used to usher the provisioning process along as if terraform has run, and consequently
// the timing data is extracted from the log content and validated against the expected values.
func TestTimingsFromProvision(t *testing.T) {
t.Parallel()
cwd, err := os.Getwd()
require.NoError(t, err)
// Given: a fake terraform bin that behaves as we expect it to.
fakeBin := filepath.Join(cwd, "testdata", "timings-aggregation/fake-terraform.sh")
t.Log(fakeBin)
ctx, api := setupProvisioner(t, &provisionerServeOptions{
binaryPath: fakeBin,
})
sess := configure(ctx, t, api, &proto.Config{
TemplateSourceArchive: testutil.CreateTar(t, nil),
})
ctx, cancel := context.WithTimeout(ctx, testutil.WaitLong)
t.Cleanup(cancel)
// When: a plan is executed in the provisioner, our fake terraform will be executed and will produce a
// state file and some log content.
err = sendPlan(sess, proto.WorkspaceTransition_START)
require.NoError(t, err)
var timings []*proto.Timing
for {
select {
case <-ctx.Done():
t.Fatal(ctx.Err())
default:
}
msg, err := sess.Recv()
require.NoError(t, err)
if log := msg.GetLog(); log != nil {
t.Logf("%s: %s: %s", "plan", log.Level.String(), log.Output)
}
if c := msg.GetPlan(); c != nil {
require.Empty(t, c.Error)
// Capture the timing information returned by the plan process.
timings = append(timings, c.GetTimings()...)
break
}
}
// When: the plan has completed, let's trigger an apply.
err = sendApply(sess, proto.WorkspaceTransition_START)
require.NoError(t, err)
for {
select {
case <-ctx.Done():
t.Fatal(ctx.Err())
default:
}
msg, err := sess.Recv()
require.NoError(t, err)
if log := msg.GetLog(); log != nil {
t.Logf("%s: %s: %s", "apply", log.Level.String(), log.Output)
}
if c := msg.GetApply(); c != nil {
require.Empty(t, c.Error)
// Capture the timing information returned by the apply process.
timings = append(timings, c.GetTimings()...)
break
}
}
// Sort the timings stably to keep reduce flakiness.
terraform_internal.StableSortTimings(t, timings)
// Then: the received timings should match the expected values below.
// NOTE: These timings have been encoded to JSON format to make the tests more readable.
planTimings := terraform_internal.ParseTimingLines(t, []byte(`{"start":"2024-08-15T08:26:39.194726Z", "end":"2024-08-15T08:26:39.195836Z", "action":"read", "source":"coder", "resource":"data.coder_parameter.memory_size", "stage":"plan", "state":"COMPLETED"}
{"start":"2024-08-15T08:26:39.194726Z", "end":"2024-08-15T08:26:39.195712Z", "action":"read", "source":"coder", "resource":"data.coder_provisioner.me", "stage":"plan", "state":"COMPLETED"}
{"start":"2024-08-15T08:26:39.194726Z", "end":"2024-08-15T08:26:39.195820Z", "action":"read", "source":"coder", "resource":"data.coder_workspace.me", "stage":"plan", "state":"COMPLETED"}`))
applyTimings := terraform_internal.ParseTimingLines(t, []byte(`{"start":"2024-08-15T08:26:39.616546Z", "end":"2024-08-15T08:26:39.618045Z", "action":"create", "source":"coder", "resource":"coder_agent.main", "stage":"apply", "state":"COMPLETED"}
{"start":"2024-08-15T08:26:39.626722Z", "end":"2024-08-15T08:26:39.669954Z", "action":"create", "source":"docker", "resource":"docker_image.main", "stage":"apply", "state":"COMPLETED"}
{"start":"2024-08-15T08:26:39.627335Z", "end":"2024-08-15T08:26:39.660616Z", "action":"create", "source":"docker", "resource":"docker_volume.home_volume", "stage":"apply", "state":"COMPLETED"}
{"start":"2024-08-15T08:26:39.682223Z", "end":"2024-08-15T08:26:40.186482Z", "action":"create", "source":"docker", "resource":"docker_container.workspace[0]", "stage":"apply", "state":"COMPLETED"}`))
initTiming := terraform_internal.ParseTimingLines(t, []byte(`{"start":"2000-01-01T01:01:01.123456Z", "end":"2000-01-01T01:01:01.123456Z", "action":"initializing terraform", "source":"terraform", "resource":"state file", "stage":"init", "state":"COMPLETED"}`))[0]
graphTiming := terraform_internal.ParseTimingLines(t, []byte(`{"start":"2000-01-01T01:01:01.123456Z", "end":"2000-01-01T01:01:01.123456Z", "action":"building terraform dependency graph", "source":"terraform", "resource":"state file", "stage":"graph", "state":"COMPLETED"}`))[0]
require.Len(t, timings, len(planTimings)+len(applyTimings)+2)
// init/graph timings are computed dynamically during provisioning whereas plan/apply come from the logs (fixtures) in
// provisioner/terraform/testdata/timings-aggregation/fake-terraform.sh.
//
// This walks the timings, keeping separate cursors for plan and apply.
// We manually override the init/graph timings' timestamps so that the equality check works (all other fields should be as expected).
pCursor := 0
aCursor := 0
for _, tim := range timings {
switch tim.Stage {
case string(database.ProvisionerJobTimingStageInit):
tim.Start, tim.End = initTiming.Start, initTiming.End
require.True(t, terraform_internal.TimingsAreEqual(t, []*proto.Timing{initTiming}, []*proto.Timing{tim}))
case string(database.ProvisionerJobTimingStageGraph):
tim.Start, tim.End = graphTiming.Start, graphTiming.End
require.True(t, terraform_internal.TimingsAreEqual(t, []*proto.Timing{graphTiming}, []*proto.Timing{tim}))
case string(database.ProvisionerJobTimingStagePlan):
require.True(t, terraform_internal.TimingsAreEqual(t, []*proto.Timing{planTimings[pCursor]}, []*proto.Timing{tim}))
pCursor++
case string(database.ProvisionerJobTimingStageApply):
require.True(t, terraform_internal.TimingsAreEqual(t, []*proto.Timing{applyTimings[aCursor]}, []*proto.Timing{tim}))
aCursor++
}
}
}