| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134 |
- /*
- Copyright 2023 Docker Compose CLI authors
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package oci
- import (
- "context"
- "io"
- "net/url"
- "strings"
- "github.com/containerd/containerd/v2/core/remotes"
- "github.com/containerd/containerd/v2/core/remotes/docker"
- "github.com/containerd/containerd/v2/pkg/labels"
- "github.com/containerd/errdefs"
- "github.com/distribution/reference"
- "github.com/docker/cli/cli/config/configfile"
- "github.com/docker/compose/v2/internal/registry"
- "github.com/moby/buildkit/util/contentutil"
- spec "github.com/opencontainers/image-spec/specs-go/v1"
- )
- // NewResolver setup an OCI Resolver based on docker/cli config to provide registry credentials
- func NewResolver(config *configfile.ConfigFile) remotes.Resolver {
- return docker.NewResolver(docker.ResolverOptions{
- Hosts: docker.ConfigureDefaultRegistries(
- docker.WithAuthorizer(docker.NewDockerAuthorizer(
- docker.WithAuthCreds(func(host string) (string, string, error) {
- host = registry.GetAuthConfigKey(host)
- auth, err := config.GetAuthConfig(host)
- if err != nil {
- return "", "", err
- }
- if auth.IdentityToken != "" {
- return "", auth.IdentityToken, nil
- }
- return auth.Username, auth.Password, nil
- }),
- )),
- ),
- })
- }
- // Get retrieves a Named OCI resource and returns OCI Descriptor and Manifest
- func Get(ctx context.Context, resolver remotes.Resolver, ref reference.Named) (spec.Descriptor, []byte, error) {
- _, descriptor, err := resolver.Resolve(ctx, ref.String())
- if err != nil {
- return spec.Descriptor{}, nil, err
- }
- fetcher, err := resolver.Fetcher(ctx, ref.String())
- if err != nil {
- return spec.Descriptor{}, nil, err
- }
- fetch, err := fetcher.Fetch(ctx, descriptor)
- if err != nil {
- return spec.Descriptor{}, nil, err
- }
- content, err := io.ReadAll(fetch)
- if err != nil {
- return spec.Descriptor{}, nil, err
- }
- return descriptor, content, nil
- }
- func Copy(ctx context.Context, resolver remotes.Resolver, image reference.Named, named reference.Named) (spec.Descriptor, error) {
- src, desc, err := resolver.Resolve(ctx, image.String())
- if err != nil {
- return spec.Descriptor{}, err
- }
- if desc.Annotations == nil {
- desc.Annotations = make(map[string]string)
- }
- // set LabelDistributionSource so push will actually use a registry mount
- refspec := reference.TrimNamed(image).String()
- u, err := url.Parse("dummy://" + refspec)
- if err != nil {
- return spec.Descriptor{}, err
- }
- source, repo := u.Hostname(), strings.TrimPrefix(u.Path, "/")
- desc.Annotations[labels.LabelDistributionSource+"."+source] = repo
- p, err := resolver.Pusher(ctx, named.Name())
- if err != nil {
- return spec.Descriptor{}, err
- }
- f, err := resolver.Fetcher(ctx, src)
- if err != nil {
- return spec.Descriptor{}, err
- }
- err = contentutil.CopyChain(ctx,
- contentutil.FromPusher(p),
- contentutil.FromFetcher(f), desc)
- return desc, err
- }
- func Push(ctx context.Context, resolver remotes.Resolver, ref reference.Named, descriptor spec.Descriptor) error {
- pusher, err := resolver.Pusher(ctx, ref.String())
- if err != nil {
- return err
- }
- ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeYAMLMediaType, "artifact-")
- ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeEnvFileMediaType, "artifact-")
- ctx = remotes.WithMediaTypeKeyPrefix(ctx, ComposeEmptyConfigMediaType, "config-")
- ctx = remotes.WithMediaTypeKeyPrefix(ctx, spec.MediaTypeEmptyJSON, "config-")
- push, err := pusher.Push(ctx, descriptor)
- if errdefs.IsAlreadyExists(err) {
- return nil
- }
- if err != nil {
- return err
- }
- defer func() {
- _ = push.Close()
- }()
- _, err = push.Write(descriptor.Data)
- return err
- }
|