build.go 13 KB

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