build.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. /*
  2. Copyright 2020 Docker Compose CLI authors
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package compose
  14. import (
  15. "context"
  16. "fmt"
  17. "os"
  18. "github.com/compose-spec/compose-go/types"
  19. "github.com/containerd/containerd/platforms"
  20. "github.com/docker/buildx/build"
  21. "github.com/docker/buildx/driver"
  22. _ "github.com/docker/buildx/driver/docker" // required to get default driver registered
  23. "github.com/docker/buildx/util/buildflags"
  24. xprogress "github.com/docker/buildx/util/progress"
  25. bclient "github.com/moby/buildkit/client"
  26. "github.com/moby/buildkit/session"
  27. "github.com/moby/buildkit/session/auth/authprovider"
  28. specs "github.com/opencontainers/image-spec/specs-go/v1"
  29. "github.com/docker/compose-cli/pkg/api"
  30. "github.com/docker/compose-cli/pkg/progress"
  31. "github.com/docker/compose-cli/pkg/utils"
  32. )
  33. func (s *composeService) Build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  34. return progress.Run(ctx, func(ctx context.Context) error {
  35. return s.build(ctx, project, options)
  36. })
  37. }
  38. func (s *composeService) build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  39. opts := map[string]build.Options{}
  40. imagesToBuild := []string{}
  41. args := flatten(options.Args.Resolve(func(s string) (string, bool) {
  42. s, ok := project.Environment[s]
  43. return s, ok
  44. }))
  45. for _, service := range project.Services {
  46. if service.Build != nil {
  47. imageName := getImageName(service, project.Name)
  48. imagesToBuild = append(imagesToBuild, imageName)
  49. buildOptions, err := s.toBuildOptions(project, service, imageName)
  50. if err != nil {
  51. return err
  52. }
  53. buildOptions.Pull = options.Pull
  54. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, args)
  55. buildOptions.NoCache = options.NoCache
  56. opts[imageName] = buildOptions
  57. buildOptions.CacheFrom, err = buildflags.ParseCacheEntry(service.Build.CacheFrom)
  58. if err != nil {
  59. return err
  60. }
  61. for _, image := range service.Build.CacheFrom {
  62. buildOptions.CacheFrom = append(buildOptions.CacheFrom, bclient.CacheOptionsEntry{
  63. Type: "registry",
  64. Attrs: map[string]string{"ref": image},
  65. })
  66. }
  67. }
  68. }
  69. _, err := s.doBuild(ctx, project, opts, options.Progress)
  70. if err == nil {
  71. if len(imagesToBuild) > 0 && !options.Quiet {
  72. utils.DisplayScanSuggestMsg()
  73. }
  74. }
  75. return err
  76. }
  77. func (s *composeService) ensureImagesExists(ctx context.Context, project *types.Project, quietPull bool) error {
  78. for _, service := range project.Services {
  79. if service.Image == "" && service.Build == nil {
  80. return fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  81. }
  82. }
  83. images, err := s.getLocalImagesDigests(ctx, project)
  84. if err != nil {
  85. return err
  86. }
  87. err = s.pullRequiredImages(ctx, project, images, quietPull)
  88. if err != nil {
  89. return err
  90. }
  91. mode := xprogress.PrinterModeAuto
  92. if quietPull {
  93. mode = xprogress.PrinterModeQuiet
  94. }
  95. opts, err := s.getBuildOptions(project, images)
  96. if err != nil {
  97. return err
  98. }
  99. builtImages, err := s.doBuild(ctx, project, opts, mode)
  100. if err != nil {
  101. return err
  102. }
  103. if len(builtImages) > 0 {
  104. utils.DisplayScanSuggestMsg()
  105. }
  106. for name, digest := range builtImages {
  107. images[name] = digest
  108. }
  109. // set digest as com.docker.compose.image label so we can detect outdated containers
  110. for i, service := range project.Services {
  111. image := getImageName(service, project.Name)
  112. digest, ok := images[image]
  113. if ok {
  114. if project.Services[i].Labels == nil {
  115. project.Services[i].Labels = types.Labels{}
  116. }
  117. project.Services[i].Labels[api.ImageDigestLabel] = digest
  118. project.Services[i].Image = image
  119. }
  120. }
  121. return nil
  122. }
  123. func (s *composeService) getBuildOptions(project *types.Project, images map[string]string) (map[string]build.Options, error) {
  124. opts := map[string]build.Options{}
  125. for _, service := range project.Services {
  126. if service.Image == "" && service.Build == nil {
  127. return nil, fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  128. }
  129. imageName := getImageName(service, project.Name)
  130. _, localImagePresent := images[imageName]
  131. if service.Build != nil {
  132. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  133. continue
  134. }
  135. opt, err := s.toBuildOptions(project, service, imageName)
  136. if err != nil {
  137. return nil, err
  138. }
  139. opts[imageName] = opt
  140. continue
  141. }
  142. }
  143. return opts, nil
  144. }
  145. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  146. imageNames := []string{}
  147. for _, s := range project.Services {
  148. imgName := getImageName(s, project.Name)
  149. if !utils.StringContains(imageNames, imgName) {
  150. imageNames = append(imageNames, imgName)
  151. }
  152. }
  153. imgs, err := s.getImages(ctx, imageNames)
  154. if err != nil {
  155. return nil, err
  156. }
  157. images := map[string]string{}
  158. for name, info := range imgs {
  159. images[name] = info.ID
  160. }
  161. return images, nil
  162. }
  163. func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, mode string) (map[string]string, error) {
  164. info, err := s.apiClient.Info(ctx)
  165. if err != nil {
  166. return nil, err
  167. }
  168. if info.OSType == "windows" {
  169. // no support yet for Windows container builds in Buildkit
  170. // https://docs.docker.com/develop/develop-images/build_enhancements/#limitations
  171. err := s.windowsBuild(opts, mode)
  172. return nil, WrapCategorisedComposeError(err, BuildFailure)
  173. }
  174. if len(opts) == 0 {
  175. return nil, nil
  176. }
  177. const drivername = "default"
  178. d, err := driver.GetDriver(ctx, drivername, nil, s.apiClient, s.configFile, nil, nil, "", nil, nil, project.WorkingDir)
  179. if err != nil {
  180. return nil, err
  181. }
  182. driverInfo := []build.DriverInfo{
  183. {
  184. Name: "default",
  185. Driver: d,
  186. },
  187. }
  188. // Progress needs its own context that lives longer than the
  189. // build one otherwise it won't read all the messages from
  190. // build and will lock
  191. progressCtx, cancel := context.WithCancel(context.Background())
  192. defer cancel()
  193. w := xprogress.NewPrinter(progressCtx, os.Stdout, mode)
  194. // We rely on buildx "docker" builder integrated in docker engine, so don't need a DockerAPI here
  195. response, err := build.Build(ctx, driverInfo, opts, nil, nil, w)
  196. errW := w.Wait()
  197. if err == nil {
  198. err = errW
  199. }
  200. if err != nil {
  201. return nil, WrapCategorisedComposeError(err, BuildFailure)
  202. }
  203. imagesBuilt := map[string]string{}
  204. for name, img := range response {
  205. if img == nil || len(img.ExporterResponse) == 0 {
  206. continue
  207. }
  208. digest, ok := img.ExporterResponse["containerimage.digest"]
  209. if !ok {
  210. continue
  211. }
  212. imagesBuilt[name] = digest
  213. }
  214. return imagesBuilt, err
  215. }
  216. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, imageTag string) (build.Options, error) {
  217. var tags []string
  218. tags = append(tags, imageTag)
  219. buildArgs := flatten(service.Build.Args.Resolve(func(s string) (string, bool) {
  220. s, ok := project.Environment[s]
  221. return s, ok
  222. }))
  223. var plats []specs.Platform
  224. if service.Platform != "" {
  225. p, err := platforms.Parse(service.Platform)
  226. if err != nil {
  227. return build.Options{}, err
  228. }
  229. plats = append(plats, p)
  230. }
  231. return build.Options{
  232. Inputs: build.Inputs{
  233. ContextPath: service.Build.Context,
  234. DockerfilePath: service.Build.Dockerfile,
  235. },
  236. BuildArgs: buildArgs,
  237. Tags: tags,
  238. Target: service.Build.Target,
  239. Exports: []bclient.ExportEntry{{Type: "image", Attrs: map[string]string{}}},
  240. Platforms: plats,
  241. Labels: service.Build.Labels,
  242. Session: []session.Attachable{
  243. authprovider.NewDockerAuthProvider(os.Stderr),
  244. },
  245. }, nil
  246. }
  247. func flatten(in types.MappingWithEquals) types.Mapping {
  248. if len(in) == 0 {
  249. return nil
  250. }
  251. out := types.Mapping{}
  252. for k, v := range in {
  253. if v == nil {
  254. continue
  255. }
  256. out[k] = *v
  257. }
  258. return out
  259. }
  260. func mergeArgs(m ...types.Mapping) types.Mapping {
  261. merged := types.Mapping{}
  262. for _, mapping := range m {
  263. for key, val := range mapping {
  264. merged[key] = val
  265. }
  266. }
  267. return merged
  268. }