build.go 13 KB

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