build.go 13 KB

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