backend.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. package build // import "github.com/docker/docker/api/server/backend/build"
  2. import (
  3. "context"
  4. "fmt"
  5. "github.com/docker/distribution/reference"
  6. "github.com/docker/docker/api/types"
  7. "github.com/docker/docker/api/types/backend"
  8. "github.com/docker/docker/builder"
  9. buildkit "github.com/docker/docker/builder/builder-next"
  10. "github.com/docker/docker/builder/fscache"
  11. "github.com/docker/docker/image"
  12. "github.com/docker/docker/pkg/stringid"
  13. "github.com/pkg/errors"
  14. "golang.org/x/sync/errgroup"
  15. "google.golang.org/grpc"
  16. )
  17. // ImageComponent provides an interface for working with images
  18. type ImageComponent interface {
  19. SquashImage(from string, to string) (string, error)
  20. TagImageWithReference(image.ID, reference.Named) error
  21. }
  22. // Builder defines interface for running a build
  23. type Builder interface {
  24. Build(context.Context, backend.BuildConfig) (*builder.Result, error)
  25. }
  26. // Backend provides build functionality to the API router
  27. type Backend struct {
  28. builder Builder
  29. fsCache *fscache.FSCache
  30. imageComponent ImageComponent
  31. buildkit *buildkit.Builder
  32. }
  33. // NewBackend creates a new build backend from components
  34. func NewBackend(components ImageComponent, builder Builder, fsCache *fscache.FSCache, buildkit *buildkit.Builder) (*Backend, error) {
  35. return &Backend{imageComponent: components, builder: builder, fsCache: fsCache, buildkit: buildkit}, nil
  36. }
  37. // RegisterGRPC registers buildkit controller to the grpc server.
  38. func (b *Backend) RegisterGRPC(s *grpc.Server) {
  39. if b.buildkit != nil {
  40. b.buildkit.RegisterGRPC(s)
  41. }
  42. }
  43. // Build builds an image from a Source
  44. func (b *Backend) Build(ctx context.Context, config backend.BuildConfig) (string, error) {
  45. options := config.Options
  46. useBuildKit := options.Version == types.BuilderBuildKit
  47. tagger, err := NewTagger(b.imageComponent, config.ProgressWriter.StdoutFormatter, options.Tags)
  48. if err != nil {
  49. return "", err
  50. }
  51. var build *builder.Result
  52. if useBuildKit {
  53. build, err = b.buildkit.Build(ctx, config)
  54. if err != nil {
  55. return "", err
  56. }
  57. } else {
  58. build, err = b.builder.Build(ctx, config)
  59. if err != nil {
  60. return "", err
  61. }
  62. }
  63. if build == nil {
  64. return "", nil
  65. }
  66. var imageID = build.ImageID
  67. if options.Squash {
  68. if imageID, err = squashBuild(build, b.imageComponent); err != nil {
  69. return "", err
  70. }
  71. if config.ProgressWriter.AuxFormatter != nil {
  72. if err = config.ProgressWriter.AuxFormatter.Emit("moby.image.id", types.BuildResult{ID: imageID}); err != nil {
  73. return "", err
  74. }
  75. }
  76. }
  77. if !useBuildKit {
  78. stdout := config.ProgressWriter.StdoutFormatter
  79. fmt.Fprintf(stdout, "Successfully built %s\n", stringid.TruncateID(imageID))
  80. }
  81. if imageID != "" {
  82. err = tagger.TagImages(image.ID(imageID))
  83. }
  84. return imageID, err
  85. }
  86. // PruneCache removes all cached build sources
  87. func (b *Backend) PruneCache(ctx context.Context, opts types.BuildCachePruneOptions) (*types.BuildCachePruneReport, error) {
  88. eg, ctx := errgroup.WithContext(ctx)
  89. var fsCacheSize uint64
  90. eg.Go(func() error {
  91. var err error
  92. fsCacheSize, err = b.fsCache.Prune(ctx)
  93. if err != nil {
  94. return errors.Wrap(err, "failed to prune fscache")
  95. }
  96. return nil
  97. })
  98. var buildCacheSize int64
  99. var cacheIDs []string
  100. eg.Go(func() error {
  101. var err error
  102. buildCacheSize, cacheIDs, err = b.buildkit.Prune(ctx, opts)
  103. if err != nil {
  104. return errors.Wrap(err, "failed to prune build cache")
  105. }
  106. return nil
  107. })
  108. if err := eg.Wait(); err != nil {
  109. return nil, err
  110. }
  111. return &types.BuildCachePruneReport{SpaceReclaimed: fsCacheSize + uint64(buildCacheSize), CachesDeleted: cacheIDs}, nil
  112. }
  113. // Cancel cancels the build by ID
  114. func (b *Backend) Cancel(ctx context.Context, id string) error {
  115. return b.buildkit.Cancel(ctx, id)
  116. }
  117. func squashBuild(build *builder.Result, imageComponent ImageComponent) (string, error) {
  118. var fromID string
  119. if build.FromImage != nil {
  120. fromID = build.FromImage.ImageID()
  121. }
  122. imageID, err := imageComponent.SquashImage(build.ImageID, fromID)
  123. if err != nil {
  124. return "", errors.Wrap(err, "error squashing image")
  125. }
  126. return imageID, nil
  127. }