build.go 13 KB

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