build.go 12 KB

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