resolver.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. /*
  2. Copyright 2023 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 oci
  14. import (
  15. "context"
  16. "io"
  17. "net/url"
  18. "os"
  19. "strings"
  20. "github.com/containerd/containerd/v2/core/remotes"
  21. "github.com/containerd/containerd/v2/core/remotes/docker"
  22. "github.com/containerd/containerd/v2/pkg/labels"
  23. "github.com/containerd/errdefs"
  24. "github.com/distribution/reference"
  25. "github.com/docker/cli/cli/config/configfile"
  26. "github.com/docker/compose/v2/internal/registry"
  27. "github.com/moby/buildkit/util/contentutil"
  28. spec "github.com/opencontainers/image-spec/specs-go/v1"
  29. )
  30. // NewResolver setup an OCI Resolver based on docker/cli config to provide registry credentials
  31. func NewResolver(config *configfile.ConfigFile) remotes.Resolver {
  32. return docker.NewResolver(docker.ResolverOptions{
  33. Hosts: docker.ConfigureDefaultRegistries(
  34. docker.WithAuthorizer(docker.NewDockerAuthorizer(
  35. docker.WithAuthCreds(func(host string) (string, string, error) {
  36. host = registry.GetAuthConfigKey(host)
  37. auth, err := config.GetAuthConfig(host)
  38. if err != nil {
  39. return "", "", err
  40. }
  41. if auth.IdentityToken != "" {
  42. return "", auth.IdentityToken, nil
  43. }
  44. return auth.Username, auth.Password, nil
  45. }),
  46. )),
  47. docker.WithPlainHTTP(func(s string) (bool, error) {
  48. // Used for testing **only**
  49. _, b := os.LookupEnv("__TEST__INSECURE__REGISTRY__")
  50. return b, nil
  51. }),
  52. ),
  53. })
  54. }
  55. // Get retrieves a Named OCI resource and returns OCI Descriptor and Manifest
  56. func Get(ctx context.Context, resolver remotes.Resolver, ref reference.Named) (spec.Descriptor, []byte, error) {
  57. _, descriptor, err := resolver.Resolve(ctx, ref.String())
  58. if err != nil {
  59. return spec.Descriptor{}, nil, err
  60. }
  61. fetcher, err := resolver.Fetcher(ctx, ref.String())
  62. if err != nil {
  63. return spec.Descriptor{}, nil, err
  64. }
  65. fetch, err := fetcher.Fetch(ctx, descriptor)
  66. if err != nil {
  67. return spec.Descriptor{}, nil, err
  68. }
  69. content, err := io.ReadAll(fetch)
  70. if err != nil {
  71. return spec.Descriptor{}, nil, err
  72. }
  73. return descriptor, content, nil
  74. }
  75. func Copy(ctx context.Context, resolver remotes.Resolver, image reference.Named, named reference.Named) (spec.Descriptor, error) {
  76. src, desc, err := resolver.Resolve(ctx, image.String())
  77. if err != nil {
  78. return spec.Descriptor{}, err
  79. }
  80. if desc.Annotations == nil {
  81. desc.Annotations = make(map[string]string)
  82. }
  83. // set LabelDistributionSource so push will actually use a registry mount
  84. refspec := reference.TrimNamed(image).String()
  85. u, err := url.Parse("dummy://" + refspec)
  86. if err != nil {
  87. return spec.Descriptor{}, err
  88. }
  89. source, repo := u.Hostname(), strings.TrimPrefix(u.Path, "/")
  90. desc.Annotations[labels.LabelDistributionSource+"."+source] = repo
  91. p, err := resolver.Pusher(ctx, named.Name())
  92. if err != nil {
  93. return spec.Descriptor{}, err
  94. }
  95. f, err := resolver.Fetcher(ctx, src)
  96. if err != nil {
  97. return spec.Descriptor{}, err
  98. }
  99. err = contentutil.CopyChain(ctx,
  100. contentutil.FromPusher(p),
  101. contentutil.FromFetcher(f), desc)
  102. return desc, err
  103. }
  104. func Push(ctx context.Context, resolver remotes.Resolver, ref reference.Named, descriptor spec.Descriptor) error {
  105. pusher, err := resolver.Pusher(ctx, ref.String())
  106. if err != nil {
  107. return err
  108. }
  109. ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeYAMLMediaType, "artifact-")
  110. ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeEnvFileMediaType, "artifact-")
  111. ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeEmptyConfigMediaType, "config-")
  112. ctx = remotes.WithMediaTypeKeyPrefix(ctx, spec.MediaTypeEmptyJSON, "config-")
  113. push, err := pusher.Push(ctx, descriptor)
  114. if errdefs.IsAlreadyExists(err) {
  115. return nil
  116. }
  117. if err != nil {
  118. return err
  119. }
  120. _, err = push.Write(descriptor.Data)
  121. if err != nil {
  122. // Close the writer on error since Commit won't be called
  123. _ = push.Close()
  124. return err
  125. }
  126. // Commit will close the writer
  127. return push.Commit(ctx, int64(len(descriptor.Data)), descriptor.Digest)
  128. }