build.go 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  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/v2/pkg/api"
  30. "github.com/docker/compose/v2/pkg/progress"
  31. "github.com/docker/compose/v2/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. services, err := project.GetServices(options.Services...)
  46. if err != nil {
  47. return err
  48. }
  49. for _, service := range services {
  50. if service.Build != nil {
  51. imageName := getImageName(service, project.Name)
  52. imagesToBuild = append(imagesToBuild, imageName)
  53. buildOptions, err := s.toBuildOptions(project, service, imageName)
  54. if err != nil {
  55. return err
  56. }
  57. buildOptions.Pull = options.Pull
  58. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, args)
  59. buildOptions.NoCache = options.NoCache
  60. opts[imageName] = buildOptions
  61. buildOptions.CacheFrom, err = buildflags.ParseCacheEntry(service.Build.CacheFrom)
  62. if err != nil {
  63. return err
  64. }
  65. for _, image := range service.Build.CacheFrom {
  66. buildOptions.CacheFrom = append(buildOptions.CacheFrom, bclient.CacheOptionsEntry{
  67. Type: "registry",
  68. Attrs: map[string]string{"ref": image},
  69. })
  70. }
  71. }
  72. }
  73. _, err = s.doBuild(ctx, project, opts, options.Progress)
  74. if err == nil {
  75. if len(imagesToBuild) > 0 && !options.Quiet {
  76. utils.DisplayScanSuggestMsg()
  77. }
  78. }
  79. return err
  80. }
  81. func (s *composeService) ensureImagesExists(ctx context.Context, project *types.Project, quietPull bool) error {
  82. for _, service := range project.Services {
  83. if service.Image == "" && service.Build == nil {
  84. return fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  85. }
  86. }
  87. images, err := s.getLocalImagesDigests(ctx, project)
  88. if err != nil {
  89. return err
  90. }
  91. err = s.pullRequiredImages(ctx, project, images, quietPull)
  92. if err != nil {
  93. return err
  94. }
  95. mode := xprogress.PrinterModeAuto
  96. if quietPull {
  97. mode = xprogress.PrinterModeQuiet
  98. }
  99. opts, err := s.getBuildOptions(project, images)
  100. if err != nil {
  101. return err
  102. }
  103. builtImages, err := s.doBuild(ctx, project, opts, mode)
  104. if err != nil {
  105. return err
  106. }
  107. if len(builtImages) > 0 {
  108. utils.DisplayScanSuggestMsg()
  109. }
  110. for name, digest := range builtImages {
  111. images[name] = digest
  112. }
  113. // set digest as com.docker.compose.image label so we can detect outdated containers
  114. for i, service := range project.Services {
  115. image := getImageName(service, project.Name)
  116. digest, ok := images[image]
  117. if ok {
  118. if project.Services[i].Labels == nil {
  119. project.Services[i].Labels = types.Labels{}
  120. }
  121. project.Services[i].Labels[api.ImageDigestLabel] = digest
  122. project.Services[i].Image = image
  123. }
  124. }
  125. return nil
  126. }
  127. func (s *composeService) getBuildOptions(project *types.Project, images map[string]string) (map[string]build.Options, error) {
  128. opts := map[string]build.Options{}
  129. for _, service := range project.Services {
  130. if service.Image == "" && service.Build == nil {
  131. return nil, fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  132. }
  133. imageName := getImageName(service, project.Name)
  134. _, localImagePresent := images[imageName]
  135. if service.Build != nil {
  136. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  137. continue
  138. }
  139. opt, err := s.toBuildOptions(project, service, imageName)
  140. if err != nil {
  141. return nil, err
  142. }
  143. opts[imageName] = opt
  144. continue
  145. }
  146. }
  147. return opts, nil
  148. }
  149. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  150. imageNames := []string{}
  151. for _, s := range project.Services {
  152. imgName := getImageName(s, project.Name)
  153. if !utils.StringContains(imageNames, imgName) {
  154. imageNames = append(imageNames, imgName)
  155. }
  156. }
  157. imgs, err := s.getImages(ctx, imageNames)
  158. if err != nil {
  159. return nil, err
  160. }
  161. images := map[string]string{}
  162. for name, info := range imgs {
  163. images[name] = info.ID
  164. }
  165. return images, nil
  166. }
  167. func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, mode string) (map[string]string, error) {
  168. info, err := s.apiClient.Info(ctx)
  169. if err != nil {
  170. return nil, err
  171. }
  172. if info.OSType == "windows" {
  173. // no support yet for Windows container builds in Buildkit
  174. // https://docs.docker.com/develop/develop-images/build_enhancements/#limitations
  175. err := s.windowsBuild(opts, mode)
  176. return nil, WrapCategorisedComposeError(err, BuildFailure)
  177. }
  178. if len(opts) == 0 {
  179. return nil, nil
  180. }
  181. const drivername = "default"
  182. d, err := driver.GetDriver(ctx, drivername, nil, s.apiClient, s.configFile, nil, nil, "", nil, nil, project.WorkingDir)
  183. if err != nil {
  184. return nil, err
  185. }
  186. driverInfo := []build.DriverInfo{
  187. {
  188. Name: "default",
  189. Driver: d,
  190. },
  191. }
  192. // Progress needs its own context that lives longer than the
  193. // build one otherwise it won't read all the messages from
  194. // build and will lock
  195. progressCtx, cancel := context.WithCancel(context.Background())
  196. defer cancel()
  197. w := xprogress.NewPrinter(progressCtx, os.Stdout, mode)
  198. // We rely on buildx "docker" builder integrated in docker engine, so don't need a DockerAPI here
  199. response, err := build.Build(ctx, driverInfo, opts, nil, nil, w)
  200. errW := w.Wait()
  201. if err == nil {
  202. err = errW
  203. }
  204. if err != nil {
  205. return nil, WrapCategorisedComposeError(err, BuildFailure)
  206. }
  207. imagesBuilt := map[string]string{}
  208. for name, img := range response {
  209. if img == nil || len(img.ExporterResponse) == 0 {
  210. continue
  211. }
  212. digest, ok := img.ExporterResponse["containerimage.digest"]
  213. if !ok {
  214. continue
  215. }
  216. imagesBuilt[name] = digest
  217. }
  218. return imagesBuilt, err
  219. }
  220. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, imageTag string) (build.Options, error) {
  221. var tags []string
  222. tags = append(tags, imageTag)
  223. buildArgs := flatten(service.Build.Args.Resolve(func(s string) (string, bool) {
  224. s, ok := project.Environment[s]
  225. return s, ok
  226. }))
  227. var plats []specs.Platform
  228. if service.Platform != "" {
  229. p, err := platforms.Parse(service.Platform)
  230. if err != nil {
  231. return build.Options{}, err
  232. }
  233. plats = append(plats, p)
  234. }
  235. return build.Options{
  236. Inputs: build.Inputs{
  237. ContextPath: service.Build.Context,
  238. DockerfilePath: service.Build.Dockerfile,
  239. },
  240. BuildArgs: buildArgs,
  241. Tags: tags,
  242. Target: service.Build.Target,
  243. Exports: []bclient.ExportEntry{{Type: "image", Attrs: map[string]string{}}},
  244. Platforms: plats,
  245. Labels: service.Build.Labels,
  246. NetworkMode: service.Build.Network,
  247. ExtraHosts: service.Build.ExtraHosts,
  248. Session: []session.Attachable{
  249. authprovider.NewDockerAuthProvider(os.Stderr),
  250. },
  251. }, nil
  252. }
  253. func flatten(in types.MappingWithEquals) types.Mapping {
  254. if len(in) == 0 {
  255. return nil
  256. }
  257. out := types.Mapping{}
  258. for k, v := range in {
  259. if v == nil {
  260. continue
  261. }
  262. out[k] = *v
  263. }
  264. return out
  265. }
  266. func mergeArgs(m ...types.Mapping) types.Mapping {
  267. merged := types.Mapping{}
  268. for _, mapping := range m {
  269. for key, val := range mapping {
  270. merged[key] = val
  271. }
  272. }
  273. return merged
  274. }