build.go 13 KB

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