build.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  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. "os"
  18. "path/filepath"
  19. "github.com/docker/compose/v2/internal/tracing"
  20. "github.com/docker/buildx/controller/pb"
  21. "github.com/compose-spec/compose-go/types"
  22. "github.com/containerd/containerd/platforms"
  23. "github.com/docker/buildx/build"
  24. _ "github.com/docker/buildx/driver/docker" // required to get default driver registered
  25. "github.com/docker/buildx/store/storeutil"
  26. "github.com/docker/buildx/util/buildflags"
  27. xprogress "github.com/docker/buildx/util/progress"
  28. "github.com/docker/docker/builder/remotecontext/urlutil"
  29. bclient "github.com/moby/buildkit/client"
  30. "github.com/moby/buildkit/session"
  31. "github.com/moby/buildkit/session/auth/authprovider"
  32. "github.com/moby/buildkit/session/secrets/secretsprovider"
  33. "github.com/moby/buildkit/session/sshforward/sshprovider"
  34. "github.com/moby/buildkit/util/entitlements"
  35. specs "github.com/opencontainers/image-spec/specs-go/v1"
  36. "github.com/pkg/errors"
  37. "github.com/sirupsen/logrus"
  38. "github.com/docker/compose/v2/pkg/api"
  39. "github.com/docker/compose/v2/pkg/progress"
  40. "github.com/docker/compose/v2/pkg/utils"
  41. )
  42. func (s *composeService) Build(ctx context.Context, project *types.Project, options api.BuildOptions) error {
  43. err := options.Apply(project)
  44. if err != nil {
  45. return err
  46. }
  47. return progress.RunWithTitle(ctx, func(ctx context.Context) error {
  48. _, err := s.build(ctx, project, options)
  49. return err
  50. }, s.stdinfo(), "Building")
  51. }
  52. func (s *composeService) build(ctx context.Context, project *types.Project, options api.BuildOptions) (map[string]string, error) { //nolint:gocyclo
  53. args := options.Args.Resolve(envResolver(project.Environment))
  54. buildkitEnabled, err := s.dockerCli.BuildKitEnabled()
  55. if err != nil {
  56. return nil, err
  57. }
  58. // Progress needs its own context that lives longer than the
  59. // build one otherwise it won't read all the messages from
  60. // build and will lock
  61. progressCtx, cancel := context.WithCancel(context.Background())
  62. defer cancel()
  63. w, err := xprogress.NewPrinter(progressCtx, s.stdout(), os.Stdout, options.Progress)
  64. if err != nil {
  65. return nil, err
  66. }
  67. builtDigests := make([]string, len(project.Services))
  68. err = InDependencyOrder(ctx, project, func(ctx context.Context, name string) error {
  69. if len(options.Services) > 0 && !utils.Contains(options.Services, name) {
  70. return nil
  71. }
  72. service, idx := getServiceIndex(project, name)
  73. if service.Build == nil {
  74. return nil
  75. }
  76. if !buildkitEnabled {
  77. if service.Build.Args == nil {
  78. service.Build.Args = args
  79. } else {
  80. service.Build.Args = service.Build.Args.OverrideBy(args)
  81. }
  82. id, err := s.doBuildClassic(ctx, project.Name, service, options)
  83. if err != nil {
  84. return err
  85. }
  86. builtDigests[idx] = id
  87. if options.Push {
  88. return s.push(ctx, project, api.PushOptions{})
  89. }
  90. return nil
  91. }
  92. if options.Memory != 0 {
  93. fmt.Fprintln(s.stderr(), "WARNING: --memory is not supported by BuildKit and will be ignored.")
  94. }
  95. buildOptions, err := s.toBuildOptions(project, service, options)
  96. if err != nil {
  97. return err
  98. }
  99. buildOptions.BuildArgs = mergeArgs(buildOptions.BuildArgs, flatten(args))
  100. digest, err := s.doBuildBuildkit(ctx, service.Name, buildOptions, w, options.Builder)
  101. if err != nil {
  102. return err
  103. }
  104. builtDigests[idx] = digest
  105. return nil
  106. }, func(traversal *graphTraversal) {
  107. traversal.maxConcurrency = s.maxConcurrency
  108. })
  109. // enforce all build event get consumed
  110. if errw := w.Wait(); errw != nil {
  111. return nil, errw
  112. }
  113. if err != nil {
  114. return nil, err
  115. }
  116. imageIDs := map[string]string{}
  117. for i, imageDigest := range builtDigests {
  118. if imageDigest != "" {
  119. imageRef := api.GetImageNameOrDefault(project.Services[i], project.Name)
  120. imageIDs[imageRef] = imageDigest
  121. }
  122. }
  123. return imageIDs, err
  124. }
  125. func getServiceIndex(project *types.Project, name string) (types.ServiceConfig, int) {
  126. var service types.ServiceConfig
  127. var idx int
  128. for i, s := range project.Services {
  129. if s.Name == name {
  130. idx, service = i, s
  131. break
  132. }
  133. }
  134. return service, idx
  135. }
  136. func (s *composeService) ensureImagesExists(ctx context.Context, project *types.Project, quietPull bool) error {
  137. for _, service := range project.Services {
  138. if service.Image == "" && service.Build == nil {
  139. return fmt.Errorf("invalid service %q. Must specify either image or build", service.Name)
  140. }
  141. }
  142. images, err := s.getLocalImagesDigests(ctx, project)
  143. if err != nil {
  144. return err
  145. }
  146. err = tracing.SpanWrapFunc("project/pull", tracing.ProjectOptions(project),
  147. func(ctx context.Context) error {
  148. return s.pullRequiredImages(ctx, project, images, quietPull)
  149. },
  150. )(ctx)
  151. if err != nil {
  152. return err
  153. }
  154. mode := xprogress.PrinterModeAuto
  155. if quietPull {
  156. mode = xprogress.PrinterModeQuiet
  157. }
  158. buildRequired, err := s.prepareProjectForBuild(project, images)
  159. if err != nil {
  160. return err
  161. }
  162. if buildRequired {
  163. err = tracing.SpanWrapFunc("project/build", tracing.ProjectOptions(project),
  164. func(ctx context.Context) error {
  165. builtImages, err := s.build(ctx, project, api.BuildOptions{
  166. Progress: mode,
  167. })
  168. if err != nil {
  169. return err
  170. }
  171. for name, digest := range builtImages {
  172. images[name] = digest
  173. }
  174. return nil
  175. },
  176. )(ctx)
  177. if err != nil {
  178. return err
  179. }
  180. }
  181. // set digest as com.docker.compose.image label so we can detect outdated containers
  182. for i, service := range project.Services {
  183. image := api.GetImageNameOrDefault(service, project.Name)
  184. digest, ok := images[image]
  185. if ok {
  186. if project.Services[i].Labels == nil {
  187. project.Services[i].Labels = types.Labels{}
  188. }
  189. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  190. }
  191. }
  192. return nil
  193. }
  194. func (s *composeService) prepareProjectForBuild(project *types.Project, images map[string]string) (bool, error) {
  195. buildRequired := false
  196. err := api.BuildOptions{}.Apply(project)
  197. if err != nil {
  198. return false, err
  199. }
  200. for i, service := range project.Services {
  201. if service.Build == nil {
  202. continue
  203. }
  204. image := api.GetImageNameOrDefault(service, project.Name)
  205. _, localImagePresent := images[image]
  206. if localImagePresent && service.PullPolicy != types.PullPolicyBuild {
  207. service.Build = nil
  208. project.Services[i] = service
  209. continue
  210. }
  211. if service.Platform == "" {
  212. // let builder to build for default platform
  213. service.Build.Platforms = nil
  214. } else {
  215. service.Build.Platforms = []string{service.Platform}
  216. }
  217. project.Services[i] = service
  218. buildRequired = true
  219. }
  220. return buildRequired, nil
  221. }
  222. func (s *composeService) getLocalImagesDigests(ctx context.Context, project *types.Project) (map[string]string, error) {
  223. var imageNames []string
  224. for _, s := range project.Services {
  225. imgName := api.GetImageNameOrDefault(s, project.Name)
  226. if !utils.StringContains(imageNames, imgName) {
  227. imageNames = append(imageNames, imgName)
  228. }
  229. }
  230. imgs, err := s.getImages(ctx, imageNames)
  231. if err != nil {
  232. return nil, err
  233. }
  234. images := map[string]string{}
  235. for name, info := range imgs {
  236. images[name] = info.ID
  237. }
  238. for i, service := range project.Services {
  239. imgName := api.GetImageNameOrDefault(service, project.Name)
  240. digest, ok := images[imgName]
  241. if !ok {
  242. continue
  243. }
  244. if service.Platform != "" {
  245. platform, err := platforms.Parse(service.Platform)
  246. if err != nil {
  247. return nil, err
  248. }
  249. inspect, _, err := s.apiClient().ImageInspectWithRaw(ctx, digest)
  250. if err != nil {
  251. return nil, err
  252. }
  253. actual := specs.Platform{
  254. Architecture: inspect.Architecture,
  255. OS: inspect.Os,
  256. Variant: inspect.Variant,
  257. }
  258. if !platforms.NewMatcher(platform).Match(actual) {
  259. return nil, errors.Errorf("image with reference %s was found but does not match the specified platform: wanted %s, actual: %s",
  260. imgName, platforms.Format(platform), platforms.Format(actual))
  261. }
  262. }
  263. project.Services[i].CustomLabels.Add(api.ImageDigestLabel, digest)
  264. }
  265. return images, nil
  266. }
  267. func (s *composeService) toBuildOptions(project *types.Project, service types.ServiceConfig, options api.BuildOptions) (build.Options, error) {
  268. buildArgs := flatten(service.Build.Args.Resolve(envResolver(project.Environment)))
  269. for k, v := range storeutil.GetProxyConfig(s.dockerCli) {
  270. if _, ok := buildArgs[k]; !ok {
  271. buildArgs[k] = v
  272. }
  273. }
  274. plats, err := addPlatforms(project, service)
  275. if err != nil {
  276. return build.Options{}, err
  277. }
  278. cacheFrom, err := buildflags.ParseCacheEntry(service.Build.CacheFrom)
  279. if err != nil {
  280. return build.Options{}, err
  281. }
  282. cacheTo, err := buildflags.ParseCacheEntry(service.Build.CacheTo)
  283. if err != nil {
  284. return build.Options{}, err
  285. }
  286. sessionConfig := []session.Attachable{
  287. authprovider.NewDockerAuthProvider(s.configFile()),
  288. }
  289. if len(options.SSHs) > 0 || len(service.Build.SSH) > 0 {
  290. sshAgentProvider, err := sshAgentProvider(append(service.Build.SSH, options.SSHs...))
  291. if err != nil {
  292. return build.Options{}, err
  293. }
  294. sessionConfig = append(sessionConfig, sshAgentProvider)
  295. }
  296. if len(service.Build.Secrets) > 0 {
  297. secretsProvider, err := addSecretsConfig(project, service)
  298. if err != nil {
  299. return build.Options{}, err
  300. }
  301. sessionConfig = append(sessionConfig, secretsProvider)
  302. }
  303. tags := []string{api.GetImageNameOrDefault(service, project.Name)}
  304. if len(service.Build.Tags) > 0 {
  305. tags = append(tags, service.Build.Tags...)
  306. }
  307. var allow []entitlements.Entitlement
  308. if service.Build.Privileged {
  309. allow = append(allow, entitlements.EntitlementSecurityInsecure)
  310. }
  311. imageLabels := getImageBuildLabels(project, service)
  312. push := options.Push && service.Image != ""
  313. exports := []bclient.ExportEntry{{
  314. Type: "docker",
  315. Attrs: map[string]string{
  316. "load": "true",
  317. "push": fmt.Sprint(push),
  318. },
  319. }}
  320. if len(service.Build.Platforms) > 1 {
  321. exports = []bclient.ExportEntry{{
  322. Type: "image",
  323. Attrs: map[string]string{
  324. "push": fmt.Sprint(push),
  325. },
  326. }}
  327. }
  328. return build.Options{
  329. Inputs: build.Inputs{
  330. ContextPath: service.Build.Context,
  331. DockerfileInline: service.Build.DockerfileInline,
  332. DockerfilePath: dockerFilePath(service.Build.Context, service.Build.Dockerfile),
  333. NamedContexts: toBuildContexts(service.Build.AdditionalContexts),
  334. },
  335. CacheFrom: pb.CreateCaches(cacheFrom),
  336. CacheTo: pb.CreateCaches(cacheTo),
  337. NoCache: service.Build.NoCache,
  338. Pull: service.Build.Pull,
  339. BuildArgs: buildArgs,
  340. Tags: tags,
  341. Target: service.Build.Target,
  342. Exports: exports,
  343. Platforms: plats,
  344. Labels: imageLabels,
  345. NetworkMode: service.Build.Network,
  346. ExtraHosts: service.Build.ExtraHosts.AsList(),
  347. Session: sessionConfig,
  348. Allow: allow,
  349. }, nil
  350. }
  351. func flatten(in types.MappingWithEquals) types.Mapping {
  352. out := types.Mapping{}
  353. if len(in) == 0 {
  354. return out
  355. }
  356. for k, v := range in {
  357. if v == nil {
  358. continue
  359. }
  360. out[k] = *v
  361. }
  362. return out
  363. }
  364. func mergeArgs(m ...types.Mapping) types.Mapping {
  365. merged := types.Mapping{}
  366. for _, mapping := range m {
  367. for key, val := range mapping {
  368. merged[key] = val
  369. }
  370. }
  371. return merged
  372. }
  373. func dockerFilePath(ctxName string, dockerfile string) string {
  374. if dockerfile == "" {
  375. return ""
  376. }
  377. if urlutil.IsGitURL(ctxName) || filepath.IsAbs(dockerfile) {
  378. return dockerfile
  379. }
  380. return filepath.Join(ctxName, dockerfile)
  381. }
  382. func sshAgentProvider(sshKeys types.SSHConfig) (session.Attachable, error) {
  383. sshConfig := make([]sshprovider.AgentConfig, 0, len(sshKeys))
  384. for _, sshKey := range sshKeys {
  385. sshConfig = append(sshConfig, sshprovider.AgentConfig{
  386. ID: sshKey.ID,
  387. Paths: []string{sshKey.Path},
  388. })
  389. }
  390. return sshprovider.NewSSHAgentProvider(sshConfig)
  391. }
  392. func addSecretsConfig(project *types.Project, service types.ServiceConfig) (session.Attachable, error) {
  393. var sources []secretsprovider.Source
  394. for _, secret := range service.Build.Secrets {
  395. config := project.Secrets[secret.Source]
  396. id := secret.Source
  397. if secret.Target != "" {
  398. id = secret.Target
  399. }
  400. switch {
  401. case config.File != "":
  402. sources = append(sources, secretsprovider.Source{
  403. ID: id,
  404. FilePath: config.File,
  405. })
  406. case config.Environment != "":
  407. sources = append(sources, secretsprovider.Source{
  408. ID: id,
  409. Env: config.Environment,
  410. })
  411. default:
  412. return nil, fmt.Errorf("build.secrets only supports environment or file-based secrets: %q", secret.Source)
  413. }
  414. if secret.UID != "" || secret.GID != "" || secret.Mode != nil {
  415. logrus.Warn("secrets `uid`, `gid` and `mode` are not supported by BuildKit, they will be ignored")
  416. }
  417. }
  418. store, err := secretsprovider.NewStore(sources)
  419. if err != nil {
  420. return nil, err
  421. }
  422. return secretsprovider.NewSecretProvider(store), nil
  423. }
  424. func addPlatforms(project *types.Project, service types.ServiceConfig) ([]specs.Platform, error) {
  425. plats, err := useDockerDefaultOrServicePlatform(project, service, false)
  426. if err != nil {
  427. return nil, err
  428. }
  429. for _, buildPlatform := range service.Build.Platforms {
  430. p, err := platforms.Parse(buildPlatform)
  431. if err != nil {
  432. return nil, err
  433. }
  434. if !utils.Contains(plats, p) {
  435. plats = append(plats, p)
  436. }
  437. }
  438. return plats, nil
  439. }
  440. func getImageBuildLabels(project *types.Project, service types.ServiceConfig) types.Labels {
  441. ret := make(types.Labels)
  442. if service.Build != nil {
  443. for k, v := range service.Build.Labels {
  444. ret.Add(k, v)
  445. }
  446. }
  447. ret.Add(api.VersionLabel, api.ComposeVersion)
  448. ret.Add(api.ProjectLabel, project.Name)
  449. ret.Add(api.ServiceLabel, service.Name)
  450. return ret
  451. }
  452. func toBuildContexts(additionalContexts types.Mapping) map[string]build.NamedContext {
  453. namedContexts := map[string]build.NamedContext{}
  454. for name, context := range additionalContexts {
  455. namedContexts[name] = build.NamedContext{Path: context}
  456. }
  457. return namedContexts
  458. }
  459. func useDockerDefaultPlatform(project *types.Project, platformList types.StringList) ([]specs.Platform, error) {
  460. var plats []specs.Platform
  461. if platform, ok := project.Environment["DOCKER_DEFAULT_PLATFORM"]; ok {
  462. if len(platformList) > 0 && !utils.StringContains(platformList, platform) {
  463. return nil, fmt.Errorf("the DOCKER_DEFAULT_PLATFORM %q value should be part of the service.build.platforms: %q", platform, platformList)
  464. }
  465. p, err := platforms.Parse(platform)
  466. if err != nil {
  467. return nil, err
  468. }
  469. plats = append(plats, p)
  470. }
  471. return plats, nil
  472. }
  473. func useDockerDefaultOrServicePlatform(project *types.Project, service types.ServiceConfig, useOnePlatform bool) ([]specs.Platform, error) {
  474. plats, err := useDockerDefaultPlatform(project, service.Build.Platforms)
  475. if (len(plats) > 0 && useOnePlatform) || err != nil {
  476. return plats, err
  477. }
  478. if service.Platform != "" {
  479. if len(service.Build.Platforms) > 0 && !utils.StringContains(service.Build.Platforms, service.Platform) {
  480. return nil, fmt.Errorf("service.platform %q should be part of the service.build.platforms: %q", service.Platform, service.Build.Platforms)
  481. }
  482. // User defined a service platform and no build platforms, so we should keep the one define on the service level
  483. p, err := platforms.Parse(service.Platform)
  484. if !utils.Contains(plats, p) {
  485. plats = append(plats, p)
  486. }
  487. return plats, err
  488. }
  489. return plats, nil
  490. }