build.go 13 KB

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