build_session_test.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. package build
  2. import (
  3. "context"
  4. "io/ioutil"
  5. "net"
  6. "net/http"
  7. "strings"
  8. "testing"
  9. "github.com/docker/docker/api/types"
  10. dclient "github.com/docker/docker/client"
  11. "github.com/docker/docker/internal/test/daemon"
  12. "github.com/docker/docker/internal/test/fakecontext"
  13. "github.com/docker/docker/internal/test/request"
  14. "github.com/moby/buildkit/session"
  15. "github.com/moby/buildkit/session/filesync"
  16. "golang.org/x/sync/errgroup"
  17. "gotest.tools/assert"
  18. is "gotest.tools/assert/cmp"
  19. "gotest.tools/skip"
  20. )
  21. func TestBuildWithSession(t *testing.T) {
  22. skip.If(t, testEnv.DaemonInfo.OSType == "windows")
  23. var client dclient.APIClient
  24. if !testEnv.DaemonInfo.ExperimentalBuild {
  25. skip.If(t, testEnv.IsRemoteDaemon, "cannot run daemon when remote daemon")
  26. d := daemon.New(t, daemon.WithExperimental)
  27. d.StartWithBusybox(t)
  28. defer d.Stop(t)
  29. client = d.NewClientT(t)
  30. } else {
  31. client = testEnv.APIClient()
  32. }
  33. dockerfile := `
  34. FROM busybox
  35. COPY file /
  36. RUN cat /file
  37. `
  38. fctx := fakecontext.New(t, "",
  39. fakecontext.WithFile("file", "some content"),
  40. )
  41. defer fctx.Close()
  42. out := testBuildWithSession(t, client, client.DaemonHost(), fctx.Dir, dockerfile)
  43. assert.Check(t, is.Contains(out, "some content"))
  44. fctx.Add("second", "contentcontent")
  45. dockerfile += `
  46. COPY second /
  47. RUN cat /second
  48. `
  49. out = testBuildWithSession(t, client, client.DaemonHost(), fctx.Dir, dockerfile)
  50. assert.Check(t, is.Equal(strings.Count(out, "Using cache"), 2))
  51. assert.Check(t, is.Contains(out, "contentcontent"))
  52. du, err := client.DiskUsage(context.TODO())
  53. assert.Check(t, err)
  54. assert.Check(t, du.BuilderSize > 10)
  55. out = testBuildWithSession(t, client, client.DaemonHost(), fctx.Dir, dockerfile)
  56. assert.Check(t, is.Equal(strings.Count(out, "Using cache"), 4))
  57. du2, err := client.DiskUsage(context.TODO())
  58. assert.Check(t, err)
  59. assert.Check(t, is.Equal(du.BuilderSize, du2.BuilderSize))
  60. // rebuild with regular tar, confirm cache still applies
  61. fctx.Add("Dockerfile", dockerfile)
  62. // FIXME(vdemeester) use sock here
  63. res, body, err := request.Do(
  64. "/build",
  65. request.Host(client.DaemonHost()),
  66. request.Method(http.MethodPost),
  67. request.RawContent(fctx.AsTarReader(t)),
  68. request.ContentType("application/x-tar"))
  69. assert.NilError(t, err)
  70. assert.Check(t, is.DeepEqual(http.StatusOK, res.StatusCode))
  71. outBytes, err := request.ReadBody(body)
  72. assert.NilError(t, err)
  73. assert.Check(t, is.Contains(string(outBytes), "Successfully built"))
  74. assert.Check(t, is.Equal(strings.Count(string(outBytes), "Using cache"), 4))
  75. _, err = client.BuildCachePrune(context.TODO(), types.BuildCachePruneOptions{All: true})
  76. assert.Check(t, err)
  77. du, err = client.DiskUsage(context.TODO())
  78. assert.Check(t, err)
  79. assert.Check(t, is.Equal(du.BuilderSize, int64(0)))
  80. }
  81. func testBuildWithSession(t *testing.T, client dclient.APIClient, daemonHost string, dir, dockerfile string) (outStr string) {
  82. ctx := context.Background()
  83. sess, err := session.NewSession(ctx, "foo1", "foo")
  84. assert.Check(t, err)
  85. fsProvider := filesync.NewFSSyncProvider([]filesync.SyncedDir{
  86. {Dir: dir},
  87. })
  88. sess.Allow(fsProvider)
  89. g, ctx := errgroup.WithContext(ctx)
  90. g.Go(func() error {
  91. return sess.Run(ctx, func(ctx context.Context, proto string, meta map[string][]string) (net.Conn, error) {
  92. return client.DialHijack(ctx, "/session", "h2c", meta)
  93. })
  94. })
  95. g.Go(func() error {
  96. // FIXME use sock here
  97. res, body, err := request.Do(
  98. "/build?remote=client-session&session="+sess.ID(),
  99. request.Host(daemonHost),
  100. request.Method(http.MethodPost),
  101. request.With(func(req *http.Request) error {
  102. req.Body = ioutil.NopCloser(strings.NewReader(dockerfile))
  103. return nil
  104. }),
  105. )
  106. if err != nil {
  107. return err
  108. }
  109. assert.Check(t, is.DeepEqual(res.StatusCode, http.StatusOK))
  110. out, err := request.ReadBody(body)
  111. assert.NilError(t, err)
  112. assert.Check(t, is.Contains(string(out), "Successfully built"))
  113. sess.Close()
  114. outStr = string(out)
  115. return nil
  116. })
  117. err = g.Wait()
  118. assert.Check(t, err)
  119. return
  120. }