build.go 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  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. "path/filepath"
  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/docker" // required to get default driver registered
  22. "github.com/docker/buildx/util/buildflags"
  23. xprogress "github.com/docker/buildx/util/progress"
  24. "github.com/docker/docker/pkg/urlutil"
  25. bclient "github.com/moby/buildkit/client"
  26. "github.com/moby/buildkit/session"
  27. "github.com/moby/buildkit/session/auth/authprovider"
  28. "github.com/moby/buildkit/session/secrets/secretsprovider"
  29. "github.com/moby/buildkit/session/sshforward/sshprovider"
  30. specs "github.com/opencontainers/image-spec/specs-go/v1"
  31. "github.com/docker/compose/v2/pkg/api"
  32. "github.com/docker/compose/v2/pkg/progress"
  33. "github.com/docker/compose/v2/pkg/utils"
  34. )
  35. func (s *composeService) Build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  36. return progress.Run(ctx, func(ctx context.Context) error {
  37. return s.build(ctx, project, options)
  38. })
  39. }
  40. func (s *composeService) build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  41. opts := map[string]build.Options{}
  42. var imagesToBuild []string
  43. args := flatten(options.Args.Resolve(envResolver(project.Environment)))
  44. services, err := project.GetServices(options.Services...)
  45. if err != nil {
  46. return err
  47. }
  48. for _, service := range services {
  49. if service.Build == nil {
  50. continue
  51. }
  52. imageName := api.GetImageNameOrDefault(service, project.Name)
  53. imagesToBuild = append(imagesToBuild, imageName)
  54. buildOptions, err := s.toBuildOptions(project, service, imageName, options.SSHs)
  55. if err != nil {
  56. return err
  57. }
  58. buildOptions.Pull = options.Pull
  59. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, args)
  60. buildOptions.NoCache = options.NoCache
  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. opts[imageName] = buildOptions
  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 := api.GetImageNameOrDefault(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].CustomLabels[api.ImageDigestLabel] = digest
  122. }
  123. }
  124. return nil
  125. }
  126. func (s *composeService) getBuildOptions(project *types.Project, images map[string]string) (map[string]build.Options, error) {
  127. opts := map[string]build.Options{}
  128. for _, service := range project.Services {
  129. if service.Image == "" && service.Build == nil {
  130. return nil, fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  131. }
  132. imageName := api.GetImageNameOrDefault(service, project.Name)
  133. _, localImagePresent := images[imageName]
  134. if service.Build != nil {
  135. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  136. continue
  137. }
  138. opt, err := s.toBuildOptions(project, service, imageName, []types.SSHKey{})
  139. if err != nil {
  140. return nil, err
  141. }
  142. opts[imageName] = opt
  143. continue
  144. }
  145. }
  146. return opts, nil
  147. }
  148. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  149. var imageNames []string
  150. for _, s := range project.Services {
  151. imgName := api.GetImageNameOrDefault(s, project.Name)
  152. if !utils.StringContains(imageNames, imgName) {
  153. imageNames = append(imageNames, imgName)
  154. }
  155. }
  156. imgs, err := s.getImages(ctx, imageNames)
  157. if err != nil {
  158. return nil, err
  159. }
  160. images := map[string]string{}
  161. for name, info := range imgs {
  162. images[name] = info.ID
  163. }
  164. for i := range project.Services {
  165. imgName := api.GetImageNameOrDefault(project.Services[i], project.Name)
  166. digest, ok := images[imgName]
  167. if ok {
  168. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  169. }
  170. }
  171. return images, nil
  172. }
  173. func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, mode string) (map[string]string, error) {
  174. if len(opts) == 0 {
  175. return nil, nil
  176. }
  177. if buildkitEnabled, err := s.dockerCli.BuildKitEnabled(); err != nil || !buildkitEnabled {
  178. return s.doBuildClassic(ctx, project, opts)
  179. }
  180. return s.doBuildBuildkit(ctx, project, opts, mode)
  181. }
  182. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, imageTag string, sshKeys []types.SSHKey) (build.Options, error) {
  183. var tags []string
  184. tags = append(tags, imageTag)
  185. buildArgs := flatten(service.Build.Args.Resolve(envResolver(project.Environment)))
  186. var plats []specs.Platform
  187. if platform, ok := project.Environment["DOCKER_DEFAULT_PLATFORM"]; ok {
  188. p, err := platforms.Parse(platform)
  189. if err != nil {
  190. return build.Options{}, err
  191. }
  192. plats = append(plats, p)
  193. }
  194. if service.Platform != "" {
  195. p, err := platforms.Parse(service.Platform)
  196. if err != nil {
  197. return build.Options{}, err
  198. }
  199. plats = append(plats, p)
  200. }
  201. cacheFrom, err := buildflags.ParseCacheEntry(service.Build.CacheFrom)
  202. if err != nil {
  203. return build.Options{}, err
  204. }
  205. cacheTo, err := buildflags.ParseCacheEntry(service.Build.CacheTo)
  206. if err != nil {
  207. return build.Options{}, err
  208. }
  209. sessionConfig := []session.Attachable{
  210. authprovider.NewDockerAuthProvider(s.stderr()),
  211. }
  212. if len(sshKeys) > 0 || len(service.Build.SSH) > 0 {
  213. sshAgentProvider, err := sshAgentProvider(append(service.Build.SSH, sshKeys...))
  214. if err != nil {
  215. return build.Options{}, err
  216. }
  217. sessionConfig = append(sessionConfig, sshAgentProvider)
  218. }
  219. if len(service.Build.Secrets) > 0 {
  220. secretsProvider, err := addSecretsConfig(project, service)
  221. if err != nil {
  222. return build.Options{}, err
  223. }
  224. sessionConfig = append(sessionConfig, secretsProvider)
  225. }
  226. if len(service.Build.Tags) > 0 {
  227. tags = append(tags, service.Build.Tags...)
  228. }
  229. return build.Options{
  230. Inputs: build.Inputs{
  231. ContextPath: service.Build.Context,
  232. DockerfilePath: dockerFilePath(service.Build.Context, service.Build.Dockerfile),
  233. },
  234. CacheFrom: cacheFrom,
  235. CacheTo: cacheTo,
  236. NoCache: service.Build.NoCache,
  237. Pull: service.Build.Pull,
  238. BuildArgs: buildArgs,
  239. Tags: tags,
  240. Target: service.Build.Target,
  241. Exports: []bclient.ExportEntry{{Type: "image", Attrs: map[string]string{}}},
  242. Platforms: plats,
  243. Labels: service.Build.Labels,
  244. NetworkMode: service.Build.Network,
  245. ExtraHosts: service.Build.ExtraHosts.AsList(),
  246. Session: sessionConfig,
  247. }, nil
  248. }
  249. func flatten(in types.MappingWithEquals) types.Mapping {
  250. if len(in) == 0 {
  251. return nil
  252. }
  253. out := types.Mapping{}
  254. for k, v := range in {
  255. if v == nil {
  256. continue
  257. }
  258. out[k] = *v
  259. }
  260. return out
  261. }
  262. func mergeArgs(m ...types.Mapping) types.Mapping {
  263. merged := types.Mapping{}
  264. for _, mapping := range m {
  265. for key, val := range mapping {
  266. merged[key] = val
  267. }
  268. }
  269. return merged
  270. }
  271. func dockerFilePath(ctxName string, dockerfile string) string {
  272. if urlutil.IsGitURL(ctxName) || filepath.IsAbs(dockerfile) {
  273. return dockerfile
  274. }
  275. return filepath.Join(ctxName, dockerfile)
  276. }
  277. func sshAgentProvider(sshKeys types.SSHConfig) (session.Attachable, error) {
  278. sshConfig := make([]sshprovider.AgentConfig, 0, len(sshKeys))
  279. for _, sshKey := range sshKeys {
  280. sshConfig = append(sshConfig, sshprovider.AgentConfig{
  281. ID: sshKey.ID,
  282. Paths: []string{sshKey.Path},
  283. })
  284. }
  285. return sshprovider.NewSSHAgentProvider(sshConfig)
  286. }
  287. func addSecretsConfig(project *types.Project, service types.ServiceConfig) (session.Attachable, error) {
  288. var sources []secretsprovider.Source
  289. for _, secret := range service.Build.Secrets {
  290. config := project.Secrets[secret.Source]
  291. switch {
  292. case config.File != "":
  293. sources = append(sources, secretsprovider.Source{
  294. ID: secret.Source,
  295. FilePath: config.File,
  296. })
  297. case config.Environment != "":
  298. sources = append(sources, secretsprovider.Source{
  299. ID: secret.Source,
  300. Env: config.Environment,
  301. })
  302. default:
  303. return nil, fmt.Errorf("build.secrets only supports environment or file-based secrets: %q", secret.Source)
  304. }
  305. }
  306. store, err := secretsprovider.NewStore(sources)
  307. if err != nil {
  308. return nil, err
  309. }
  310. return secretsprovider.NewSecretProvider(store), nil
  311. }