build_traces_test.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. package build
  2. import (
  3. "context"
  4. "fmt"
  5. "testing"
  6. "time"
  7. "github.com/docker/docker/client/buildkit"
  8. "github.com/docker/docker/testutil"
  9. moby_buildkit_v1 "github.com/moby/buildkit/api/services/control"
  10. "github.com/moby/buildkit/client"
  11. "github.com/moby/buildkit/client/llb"
  12. "github.com/moby/buildkit/util/progress/progressui"
  13. "go.opentelemetry.io/otel"
  14. "golang.org/x/sync/errgroup"
  15. "gotest.tools/v3/assert"
  16. "gotest.tools/v3/poll"
  17. "gotest.tools/v3/skip"
  18. )
  19. type testWriter struct {
  20. *testing.T
  21. }
  22. func (t *testWriter) Write(p []byte) (int, error) {
  23. t.Log(string(p))
  24. return len(p), nil
  25. }
  26. func TestBuildkitHistoryTracePropagation(t *testing.T) {
  27. skip.If(t, testEnv.DaemonInfo.OSType == "windows", "buildkit is not supported on Windows")
  28. ctx := testutil.StartSpan(baseContext, t)
  29. opts := buildkit.ClientOpts(testEnv.APIClient())
  30. bc, err := client.New(ctx, "", opts...)
  31. assert.NilError(t, err)
  32. defer bc.Close()
  33. def, err := llb.Scratch().Marshal(ctx)
  34. assert.NilError(t, err)
  35. eg, ctxGo := errgroup.WithContext(ctx)
  36. ch := make(chan *client.SolveStatus)
  37. ctxHistory, cancel := context.WithCancel(ctx)
  38. defer cancel()
  39. sub, err := bc.ControlClient().ListenBuildHistory(ctxHistory, &moby_buildkit_v1.BuildHistoryRequest{ActiveOnly: true})
  40. assert.NilError(t, err)
  41. sub.CloseSend()
  42. defer func() {
  43. cancel()
  44. <-sub.Context().Done()
  45. }()
  46. d, err := progressui.NewDisplay(&testWriter{t}, progressui.AutoMode, progressui.WithPhase("test"))
  47. assert.NilError(t, err)
  48. eg.Go(func() error {
  49. _, err := d.UpdateFrom(ctxGo, ch)
  50. return err
  51. })
  52. eg.Go(func() error {
  53. _, err := bc.Solve(ctxGo, def, client.SolveOpt{}, ch)
  54. return err
  55. })
  56. assert.NilError(t, eg.Wait())
  57. he, err := sub.Recv()
  58. assert.NilError(t, err)
  59. assert.Assert(t, he != nil)
  60. cancel()
  61. // Traces for history records are recorded asynchronously, so we need to wait for it to be available.
  62. if he.Record.Trace != nil {
  63. return
  64. }
  65. // Split this into a new span so it doesn't clutter up the trace reporting GUI.
  66. ctx, span := otel.Tracer("").Start(ctx, "Wait for trace to propagate to history record")
  67. defer span.End()
  68. t.Log("Waiting for trace to be available")
  69. poll.WaitOn(t, func(logger poll.LogT) poll.Result {
  70. ctx, cancel := context.WithCancel(ctx)
  71. defer cancel()
  72. sub, err := bc.ControlClient().ListenBuildHistory(ctx, &moby_buildkit_v1.BuildHistoryRequest{Ref: he.Record.Ref})
  73. if err != nil {
  74. return poll.Error(err)
  75. }
  76. sub.CloseSend()
  77. defer func() {
  78. cancel()
  79. <-sub.Context().Done()
  80. }()
  81. msg, err := sub.Recv()
  82. if err != nil {
  83. return poll.Error(err)
  84. }
  85. if msg.Record.Ref != he.Record.Ref {
  86. return poll.Error(fmt.Errorf("got incorrect history record"))
  87. }
  88. if msg.Record.Trace != nil {
  89. return poll.Success()
  90. }
  91. return poll.Continue("trace not available yet")
  92. }, poll.WithDelay(time.Second), poll.WithTimeout(30*time.Second))
  93. }