build.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  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/store/storeutil"
  23. "github.com/docker/buildx/util/buildflags"
  24. xprogress "github.com/docker/buildx/util/progress"
  25. "github.com/docker/docker/builder/remotecontext/urlutil"
  26. bclient "github.com/moby/buildkit/client"
  27. "github.com/moby/buildkit/session"
  28. "github.com/moby/buildkit/session/auth/authprovider"
  29. "github.com/moby/buildkit/session/secrets/secretsprovider"
  30. "github.com/moby/buildkit/session/sshforward/sshprovider"
  31. "github.com/moby/buildkit/util/entitlements"
  32. specs "github.com/opencontainers/image-spec/specs-go/v1"
  33. "github.com/docker/compose/v2/pkg/api"
  34. "github.com/docker/compose/v2/pkg/progress"
  35. "github.com/docker/compose/v2/pkg/utils"
  36. )
  37. func (s *composeService) Build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  38. return progress.Run(ctx, func(ctx context.Context) error {
  39. _, err := s.build(ctx, project, options)
  40. return err
  41. }, s.stderr())
  42. }
  43. func (s *composeService) build(ctx context.Context, project *types.Project, options api.BuildOptions) (map[string]string, error) {
  44. args := flatten(options.Args.Resolve(envResolver(project.Environment)))
  45. builtIDs := make([]string, len(project.Services))
  46. err := InDependencyOrder(ctx, project, func(ctx context.Context, name string) error {
  47. if len(options.Services) > 0 && !utils.Contains(options.Services, name) {
  48. return nil
  49. }
  50. for i, service := range project.Services {
  51. if service.Name != name {
  52. continue
  53. }
  54. service, err := project.GetService(name)
  55. if err != nil {
  56. return err
  57. }
  58. if service.Build == nil {
  59. return nil
  60. }
  61. imageName := api.GetImageNameOrDefault(service, project.Name)
  62. buildOptions, err := s.toBuildOptions(project, service, imageName, options.SSHs)
  63. if err != nil {
  64. return err
  65. }
  66. buildOptions.Pull = options.Pull
  67. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, args)
  68. buildOptions.NoCache = options.NoCache
  69. buildOptions.CacheFrom, err = buildflags.ParseCacheEntry(service.Build.CacheFrom)
  70. if err != nil {
  71. return err
  72. }
  73. if len(service.Build.AdditionalContexts) > 0 {
  74. buildOptions.Inputs.NamedContexts = toBuildContexts(service.Build.AdditionalContexts)
  75. }
  76. for _, image := range service.Build.CacheFrom {
  77. buildOptions.CacheFrom = append(buildOptions.CacheFrom, bclient.CacheOptionsEntry{
  78. Type: "registry",
  79. Attrs: map[string]string{"ref": image},
  80. })
  81. }
  82. buildOptions.Exports = []bclient.ExportEntry{{
  83. Type: "docker",
  84. Attrs: map[string]string{
  85. "load": "true",
  86. "push": fmt.Sprint(options.Push),
  87. },
  88. }}
  89. if len(buildOptions.Platforms) > 1 {
  90. buildOptions.Exports = []bclient.ExportEntry{{
  91. Type: "image",
  92. Attrs: map[string]string{
  93. "push": fmt.Sprint(options.Push),
  94. },
  95. }}
  96. }
  97. opts := map[string]build.Options{imageName: buildOptions}
  98. ids, err := s.doBuild(ctx, project, opts, options.Progress)
  99. if err != nil {
  100. return err
  101. }
  102. builtIDs[i] = ids[imageName]
  103. }
  104. return nil
  105. }, func(traversal *graphTraversal) {
  106. traversal.maxConcurrency = s.maxConcurrency
  107. })
  108. imageIDs := map[string]string{}
  109. for i, d := range builtIDs {
  110. if d != "" {
  111. imageIDs[project.Services[i].Image] = d
  112. }
  113. }
  114. return imageIDs, err
  115. }
  116. func (s *composeService) ensureImagesExists(ctx context.Context, project *types.Project, quietPull bool) error {
  117. for _, service := range project.Services {
  118. if service.Image == "" && service.Build == nil {
  119. return fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  120. }
  121. }
  122. images, err := s.getLocalImagesDigests(ctx, project)
  123. if err != nil {
  124. return err
  125. }
  126. err = s.pullRequiredImages(ctx, project, images, quietPull)
  127. if err != nil {
  128. return err
  129. }
  130. mode := xprogress.PrinterModeAuto
  131. if quietPull {
  132. mode = xprogress.PrinterModeQuiet
  133. }
  134. opts, err := s.getBuildOptions(project, images)
  135. if err != nil {
  136. return err
  137. }
  138. builtImages, err := s.doBuild(ctx, project, opts, mode)
  139. if err != nil {
  140. return err
  141. }
  142. for name, digest := range builtImages {
  143. images[name] = digest
  144. }
  145. // set digest as com.docker.compose.image label so we can detect outdated containers
  146. for i, service := range project.Services {
  147. image := api.GetImageNameOrDefault(service, project.Name)
  148. digest, ok := images[image]
  149. if ok {
  150. if project.Services[i].Labels == nil {
  151. project.Services[i].Labels = types.Labels{}
  152. }
  153. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  154. }
  155. }
  156. return nil
  157. }
  158. func (s *composeService) getBuildOptions(project *types.Project, images map[string]string) (map[string]build.Options, error) {
  159. opts := map[string]build.Options{}
  160. for _, service := range project.Services {
  161. if service.Image == "" && service.Build == nil {
  162. return nil, fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  163. }
  164. imageName := api.GetImageNameOrDefault(service, project.Name)
  165. _, localImagePresent := images[imageName]
  166. if service.Build != nil {
  167. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  168. continue
  169. }
  170. opt, err := s.toBuildOptions(project, service, imageName, []types.SSHKey{})
  171. if err != nil {
  172. return nil, err
  173. }
  174. opt.Exports = []bclient.ExportEntry{{
  175. Type: "docker",
  176. Attrs: map[string]string{
  177. "load": "true",
  178. },
  179. }}
  180. if opt.Platforms, err = useDockerDefaultOrServicePlatform(project, service, true); err != nil {
  181. opt.Platforms = []specs.Platform{}
  182. }
  183. opts[imageName] = opt
  184. continue
  185. }
  186. }
  187. return opts, nil
  188. }
  189. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  190. var imageNames []string
  191. for _, s := range project.Services {
  192. imgName := api.GetImageNameOrDefault(s, project.Name)
  193. if !utils.StringContains(imageNames, imgName) {
  194. imageNames = append(imageNames, imgName)
  195. }
  196. }
  197. imgs, err := s.getImages(ctx, imageNames)
  198. if err != nil {
  199. return nil, err
  200. }
  201. images := map[string]string{}
  202. for name, info := range imgs {
  203. images[name] = info.ID
  204. }
  205. for i := range project.Services {
  206. imgName := api.GetImageNameOrDefault(project.Services[i], project.Name)
  207. digest, ok := images[imgName]
  208. if ok {
  209. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  210. }
  211. }
  212. return images, nil
  213. }
  214. func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, mode string) (map[string]string, error) {
  215. if len(opts) == 0 {
  216. return nil, nil
  217. }
  218. if buildkitEnabled, err := s.dockerCli.BuildKitEnabled(); err != nil || !buildkitEnabled {
  219. return s.doBuildClassic(ctx, project, opts)
  220. }
  221. return s.doBuildBuildkit(ctx, opts, mode)
  222. }
  223. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, imageTag string, sshKeys []types.SSHKey) (build.Options, error) {
  224. var tags []string
  225. tags = append(tags, imageTag)
  226. buildArgs := flatten(service.Build.Args.Resolve(envResolver(project.Environment)))
  227. for k, v := range storeutil.GetProxyConfig(s.dockerCli) {
  228. if _, ok := buildArgs[k]; !ok {
  229. buildArgs[k] = v
  230. }
  231. }
  232. plats, err := addPlatforms(project, service)
  233. if err != nil {
  234. return build.Options{}, err
  235. }
  236. cacheFrom, err := buildflags.ParseCacheEntry(service.Build.CacheFrom)
  237. if err != nil {
  238. return build.Options{}, err
  239. }
  240. cacheTo, err := buildflags.ParseCacheEntry(service.Build.CacheTo)
  241. if err != nil {
  242. return build.Options{}, err
  243. }
  244. sessionConfig := []session.Attachable{
  245. authprovider.NewDockerAuthProvider(s.configFile()),
  246. }
  247. if len(sshKeys) > 0 || len(service.Build.SSH) > 0 {
  248. sshAgentProvider, err := sshAgentProvider(append(service.Build.SSH, sshKeys...))
  249. if err != nil {
  250. return build.Options{}, err
  251. }
  252. sessionConfig = append(sessionConfig, sshAgentProvider)
  253. }
  254. if len(service.Build.Secrets) > 0 {
  255. secretsProvider, err := addSecretsConfig(project, service)
  256. if err != nil {
  257. return build.Options{}, err
  258. }
  259. sessionConfig = append(sessionConfig, secretsProvider)
  260. }
  261. if len(service.Build.Tags) > 0 {
  262. tags = append(tags, service.Build.Tags...)
  263. }
  264. var allow []entitlements.Entitlement
  265. if service.Build.Privileged {
  266. allow = append(allow, entitlements.EntitlementSecurityInsecure)
  267. }
  268. imageLabels := getImageBuildLabels(project, service)
  269. return build.Options{
  270. Inputs: build.Inputs{
  271. ContextPath: service.Build.Context,
  272. DockerfileInline: service.Build.DockerfileInline,
  273. DockerfilePath: dockerFilePath(service.Build.Context, service.Build.Dockerfile),
  274. },
  275. CacheFrom: cacheFrom,
  276. CacheTo: cacheTo,
  277. NoCache: service.Build.NoCache,
  278. Pull: service.Build.Pull,
  279. BuildArgs: buildArgs,
  280. Tags: tags,
  281. Target: service.Build.Target,
  282. Exports: []bclient.ExportEntry{{Type: "image", Attrs: map[string]string{}}},
  283. Platforms: plats,
  284. Labels: imageLabels,
  285. NetworkMode: service.Build.Network,
  286. ExtraHosts: service.Build.ExtraHosts.AsList(),
  287. Session: sessionConfig,
  288. Allow: allow,
  289. }, nil
  290. }
  291. func flatten(in types.MappingWithEquals) types.Mapping {
  292. out := types.Mapping{}
  293. if len(in) == 0 {
  294. return out
  295. }
  296. for k, v := range in {
  297. if v == nil {
  298. continue
  299. }
  300. out[k] = *v
  301. }
  302. return out
  303. }
  304. func mergeArgs(m ...types.Mapping) types.Mapping {
  305. merged := types.Mapping{}
  306. for _, mapping := range m {
  307. for key, val := range mapping {
  308. merged[key] = val
  309. }
  310. }
  311. return merged
  312. }
  313. func dockerFilePath(ctxName string, dockerfile string) string {
  314. if dockerfile == "" {
  315. return ""
  316. }
  317. if urlutil.IsGitURL(ctxName) || filepath.IsAbs(dockerfile) {
  318. return dockerfile
  319. }
  320. return filepath.Join(ctxName, dockerfile)
  321. }
  322. func sshAgentProvider(sshKeys types.SSHConfig) (session.Attachable, error) {
  323. sshConfig := make([]sshprovider.AgentConfig, 0, len(sshKeys))
  324. for _, sshKey := range sshKeys {
  325. sshConfig = append(sshConfig, sshprovider.AgentConfig{
  326. ID: sshKey.ID,
  327. Paths: []string{sshKey.Path},
  328. })
  329. }
  330. return sshprovider.NewSSHAgentProvider(sshConfig)
  331. }
  332. func addSecretsConfig(project *types.Project, service types.ServiceConfig) (session.Attachable, error) {
  333. var sources []secretsprovider.Source
  334. for _, secret := range service.Build.Secrets {
  335. config := project.Secrets[secret.Source]
  336. switch {
  337. case config.File != "":
  338. sources = append(sources, secretsprovider.Source{
  339. ID: secret.Source,
  340. FilePath: config.File,
  341. })
  342. case config.Environment != "":
  343. sources = append(sources, secretsprovider.Source{
  344. ID: secret.Source,
  345. Env: config.Environment,
  346. })
  347. default:
  348. return nil, fmt.Errorf("build.secrets only supports environment or file-based secrets: %q", secret.Source)
  349. }
  350. }
  351. store, err := secretsprovider.NewStore(sources)
  352. if err != nil {
  353. return nil, err
  354. }
  355. return secretsprovider.NewSecretProvider(store), nil
  356. }
  357. func addPlatforms(project *types.Project, service types.ServiceConfig) ([]specs.Platform, error) {
  358. plats, err := useDockerDefaultOrServicePlatform(project, service, false)
  359. if err != nil {
  360. return nil, err
  361. }
  362. for _, buildPlatform := range service.Build.Platforms {
  363. p, err := platforms.Parse(buildPlatform)
  364. if err != nil {
  365. return nil, err
  366. }
  367. if !utils.Contains(plats, p) {
  368. plats = append(plats, p)
  369. }
  370. }
  371. return plats, nil
  372. }
  373. func getImageBuildLabels(project *types.Project, service types.ServiceConfig) types.Labels {
  374. ret := make(types.Labels)
  375. if service.Build != nil {
  376. for k, v := range service.Build.Labels {
  377. ret.Add(k, v)
  378. }
  379. }
  380. ret.Add(api.VersionLabel, api.ComposeVersion)
  381. ret.Add(api.ProjectLabel, project.Name)
  382. ret.Add(api.ServiceLabel, service.Name)
  383. return ret
  384. }
  385. func toBuildContexts(additionalContexts map[string]*string) map[string]build.NamedContext {
  386. namedContexts := map[string]build.NamedContext{}
  387. for name, buildContext := range additionalContexts {
  388. namedContexts[name] = build.NamedContext{Path: *buildContext}
  389. }
  390. return namedContexts
  391. }
  392. func useDockerDefaultPlatform(project *types.Project, platformList types.StringList) ([]specs.Platform, error) {
  393. var plats []specs.Platform
  394. if platform, ok := project.Environment["DOCKER_DEFAULT_PLATFORM"]; ok {
  395. if len(platformList) > 0 && !utils.StringContains(platformList, platform) {
  396. return nil, fmt.Errorf("the DOCKER_DEFAULT_PLATFORM %q value should be part of the service.build.platforms: %q", platform, platformList)
  397. }
  398. p, err := platforms.Parse(platform)
  399. if err != nil {
  400. return nil, err
  401. }
  402. plats = append(plats, p)
  403. }
  404. return plats, nil
  405. }
  406. func useDockerDefaultOrServicePlatform(project *types.Project, service types.ServiceConfig, useOnePlatform bool) ([]specs.Platform, error) {
  407. plats, err := useDockerDefaultPlatform(project, service.Build.Platforms)
  408. if (len(plats) > 0 && useOnePlatform) || err != nil {
  409. return plats, err
  410. }
  411. if service.Platform != "" {
  412. if len(service.Build.Platforms) > 0 && !utils.StringContains(service.Build.Platforms, service.Platform) {
  413. return nil, fmt.Errorf("service.platform %q should be part of the service.build.platforms: %q", service.Platform, service.Build.Platforms)
  414. }
  415. // User defined a service platform and no build platforms, so we should keep the one define on the service level
  416. p, err := platforms.Parse(service.Platform)
  417. if !utils.Contains(plats, p) {
  418. plats = append(plats, p)
  419. }
  420. return plats, err
  421. }
  422. return plats, nil
  423. }