build.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  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/builder/remotecontext/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. args := flatten(options.Args.Resolve(envResolver(project.Environment)))
  43. services, err := project.GetServices(options.Services...)
  44. if err != nil {
  45. return err
  46. }
  47. for _, service := range services {
  48. if service.Build == nil {
  49. continue
  50. }
  51. imageName := api.GetImageNameOrDefault(service, project.Name)
  52. buildOptions, err := s.toBuildOptions(project, service, imageName, options.SSHs)
  53. if err != nil {
  54. return err
  55. }
  56. buildOptions.Pull = options.Pull
  57. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, args)
  58. buildOptions.NoCache = options.NoCache
  59. buildOptions.CacheFrom, err = buildflags.ParseCacheEntry(service.Build.CacheFrom)
  60. if err != nil {
  61. return err
  62. }
  63. for _, image := range service.Build.CacheFrom {
  64. buildOptions.CacheFrom = append(buildOptions.CacheFrom, bclient.CacheOptionsEntry{
  65. Type: "registry",
  66. Attrs: map[string]string{"ref": image},
  67. })
  68. }
  69. buildOptions.Exports = []bclient.ExportEntry{{
  70. Type: "docker",
  71. Attrs: map[string]string{
  72. "load": "true",
  73. },
  74. }}
  75. if len(buildOptions.Platforms) > 1 {
  76. buildOptions.Exports = []bclient.ExportEntry{{
  77. Type: "image",
  78. Attrs: map[string]string{},
  79. }}
  80. }
  81. opts[imageName] = buildOptions
  82. }
  83. _, err = s.doBuild(ctx, project, opts, options.Progress)
  84. return err
  85. }
  86. func (s *composeService) ensureImagesExists(ctx context.Context, project *types.Project, quietPull bool) error {
  87. for _, service := range project.Services {
  88. if service.Image == "" && service.Build == nil {
  89. return fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  90. }
  91. }
  92. images, err := s.getLocalImagesDigests(ctx, project)
  93. if err != nil {
  94. return err
  95. }
  96. err = s.pullRequiredImages(ctx, project, images, quietPull)
  97. if err != nil {
  98. return err
  99. }
  100. mode := xprogress.PrinterModeAuto
  101. if quietPull {
  102. mode = xprogress.PrinterModeQuiet
  103. }
  104. opts, err := s.getBuildOptions(project, images)
  105. if err != nil {
  106. return err
  107. }
  108. builtImages, err := s.doBuild(ctx, project, opts, mode)
  109. if err != nil {
  110. return err
  111. }
  112. for name, digest := range builtImages {
  113. images[name] = digest
  114. }
  115. // set digest as com.docker.compose.image label so we can detect outdated containers
  116. for i, service := range project.Services {
  117. image := api.GetImageNameOrDefault(service, project.Name)
  118. digest, ok := images[image]
  119. if ok {
  120. if project.Services[i].Labels == nil {
  121. project.Services[i].Labels = types.Labels{}
  122. }
  123. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  124. }
  125. }
  126. return nil
  127. }
  128. func (s *composeService) getBuildOptions(project *types.Project, images map[string]string) (map[string]build.Options, error) {
  129. opts := map[string]build.Options{}
  130. for _, service := range project.Services {
  131. if service.Image == "" && service.Build == nil {
  132. return nil, fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  133. }
  134. imageName := api.GetImageNameOrDefault(service, project.Name)
  135. _, localImagePresent := images[imageName]
  136. if service.Build != nil {
  137. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  138. continue
  139. }
  140. opt, err := s.toBuildOptions(project, service, imageName, []types.SSHKey{})
  141. if err != nil {
  142. return nil, err
  143. }
  144. opt.Exports = []bclient.ExportEntry{{
  145. Type: "docker",
  146. Attrs: map[string]string{
  147. "load": "true",
  148. },
  149. }}
  150. if opt.Platforms, err = useDockerDefaultOrServicePlatform(project, service, true); err != nil {
  151. opt.Platforms = []specs.Platform{}
  152. }
  153. opts[imageName] = opt
  154. continue
  155. }
  156. }
  157. return opts, nil
  158. }
  159. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  160. var imageNames []string
  161. for _, s := range project.Services {
  162. imgName := api.GetImageNameOrDefault(s, project.Name)
  163. if !utils.StringContains(imageNames, imgName) {
  164. imageNames = append(imageNames, imgName)
  165. }
  166. }
  167. imgs, err := s.getImages(ctx, imageNames)
  168. if err != nil {
  169. return nil, err
  170. }
  171. images := map[string]string{}
  172. for name, info := range imgs {
  173. images[name] = info.ID
  174. }
  175. for i := range project.Services {
  176. imgName := api.GetImageNameOrDefault(project.Services[i], project.Name)
  177. digest, ok := images[imgName]
  178. if ok {
  179. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  180. }
  181. }
  182. return images, nil
  183. }
  184. func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, mode string) (map[string]string, error) {
  185. if len(opts) == 0 {
  186. return nil, nil
  187. }
  188. if buildkitEnabled, err := s.dockerCli.BuildKitEnabled(); err != nil || !buildkitEnabled {
  189. return s.doBuildClassic(ctx, project, opts)
  190. }
  191. return s.doBuildBuildkit(ctx, opts, mode)
  192. }
  193. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, imageTag string, sshKeys []types.SSHKey) (build.Options, error) {
  194. var tags []string
  195. tags = append(tags, imageTag)
  196. buildArgs := flatten(service.Build.Args.Resolve(envResolver(project.Environment)))
  197. plats, err := addPlatforms(project, service)
  198. if err != nil {
  199. return build.Options{}, err
  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.configFile()),
  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. imageLabels := getImageBuildLabels(project, service)
  230. return build.Options{
  231. Inputs: build.Inputs{
  232. ContextPath: service.Build.Context,
  233. DockerfilePath: dockerFilePath(service.Build.Context, service.Build.Dockerfile),
  234. },
  235. CacheFrom: cacheFrom,
  236. CacheTo: cacheTo,
  237. NoCache: service.Build.NoCache,
  238. Pull: service.Build.Pull,
  239. BuildArgs: buildArgs,
  240. Tags: tags,
  241. Target: service.Build.Target,
  242. Exports: []bclient.ExportEntry{{Type: "image", Attrs: map[string]string{}}},
  243. Platforms: plats,
  244. Labels: imageLabels,
  245. NetworkMode: service.Build.Network,
  246. ExtraHosts: service.Build.ExtraHosts.AsList(),
  247. Session: sessionConfig,
  248. }, nil
  249. }
  250. func flatten(in types.MappingWithEquals) types.Mapping {
  251. if len(in) == 0 {
  252. return nil
  253. }
  254. out := types.Mapping{}
  255. for k, v := range in {
  256. if v == nil {
  257. continue
  258. }
  259. out[k] = *v
  260. }
  261. return out
  262. }
  263. func mergeArgs(m ...types.Mapping) types.Mapping {
  264. merged := types.Mapping{}
  265. for _, mapping := range m {
  266. for key, val := range mapping {
  267. merged[key] = val
  268. }
  269. }
  270. return merged
  271. }
  272. func dockerFilePath(ctxName string, dockerfile string) string {
  273. if urlutil.IsGitURL(ctxName) || filepath.IsAbs(dockerfile) {
  274. return dockerfile
  275. }
  276. return filepath.Join(ctxName, dockerfile)
  277. }
  278. func sshAgentProvider(sshKeys types.SSHConfig) (session.Attachable, error) {
  279. sshConfig := make([]sshprovider.AgentConfig, 0, len(sshKeys))
  280. for _, sshKey := range sshKeys {
  281. sshConfig = append(sshConfig, sshprovider.AgentConfig{
  282. ID: sshKey.ID,
  283. Paths: []string{sshKey.Path},
  284. })
  285. }
  286. return sshprovider.NewSSHAgentProvider(sshConfig)
  287. }
  288. func addSecretsConfig(project *types.Project, service types.ServiceConfig) (session.Attachable, error) {
  289. var sources []secretsprovider.Source
  290. for _, secret := range service.Build.Secrets {
  291. config := project.Secrets[secret.Source]
  292. switch {
  293. case config.File != "":
  294. sources = append(sources, secretsprovider.Source{
  295. ID: secret.Source,
  296. FilePath: config.File,
  297. })
  298. case config.Environment != "":
  299. sources = append(sources, secretsprovider.Source{
  300. ID: secret.Source,
  301. Env: config.Environment,
  302. })
  303. default:
  304. return nil, fmt.Errorf("build.secrets only supports environment or file-based secrets: %q", secret.Source)
  305. }
  306. }
  307. store, err := secretsprovider.NewStore(sources)
  308. if err != nil {
  309. return nil, err
  310. }
  311. return secretsprovider.NewSecretProvider(store), nil
  312. }
  313. func addPlatforms(project *types.Project, service types.ServiceConfig) ([]specs.Platform, error) {
  314. plats, err := useDockerDefaultOrServicePlatform(project, service, false)
  315. if err != nil {
  316. return nil, err
  317. }
  318. for _, buildPlatform := range service.Build.Platforms {
  319. p, err := platforms.Parse(buildPlatform)
  320. if err != nil {
  321. return nil, err
  322. }
  323. if !utils.Contains(plats, p) {
  324. plats = append(plats, p)
  325. }
  326. }
  327. return plats, nil
  328. }
  329. func getImageBuildLabels(project *types.Project, service types.ServiceConfig) types.Labels {
  330. ret := make(types.Labels)
  331. if service.Build != nil {
  332. for k, v := range service.Build.Labels {
  333. ret.Add(k, v)
  334. }
  335. }
  336. ret.Add(api.VersionLabel, api.ComposeVersion)
  337. ret.Add(api.ProjectLabel, project.Name)
  338. ret.Add(api.ServiceLabel, service.Name)
  339. return ret
  340. }
  341. func useDockerDefaultPlatform(project *types.Project, platformList types.StringList) ([]specs.Platform, error) {
  342. var plats []specs.Platform
  343. if platform, ok := project.Environment["DOCKER_DEFAULT_PLATFORM"]; ok {
  344. if len(platformList) > 0 && !utils.StringContains(platformList, platform) {
  345. return nil, fmt.Errorf("the DOCKER_DEFAULT_PLATFORM %q value should be part of the service.build.platforms: %q", platform, platformList)
  346. }
  347. p, err := platforms.Parse(platform)
  348. if err != nil {
  349. return nil, err
  350. }
  351. plats = append(plats, p)
  352. }
  353. return plats, nil
  354. }
  355. func useDockerDefaultOrServicePlatform(project *types.Project, service types.ServiceConfig, useOnePlatform bool) ([]specs.Platform, error) {
  356. plats, err := useDockerDefaultPlatform(project, service.Build.Platforms)
  357. if (len(plats) > 0 && useOnePlatform) || err != nil {
  358. return plats, err
  359. }
  360. if service.Platform != "" {
  361. if len(service.Build.Platforms) > 0 && !utils.StringContains(service.Build.Platforms, service.Platform) {
  362. return nil, fmt.Errorf("service.platform %q should be part of the service.build.platforms: %q", service.Platform, service.Build.Platforms)
  363. }
  364. // User defined a service platform and no build platforms, so we should keep the one define on the service level
  365. p, err := platforms.Parse(service.Platform)
  366. if !utils.Contains(plats, p) {
  367. plats = append(plats, p)
  368. }
  369. return plats, err
  370. }
  371. return plats, nil
  372. }