mirror of
https://github.com/coder/coder.git
synced 2025-07-03 16:13:58 +00:00
137 lines
6.0 KiB
Go
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++
|
|
}
|
|
}
|
|
}
|