build.go 14 KB

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