mirror of
https://github.com/docker/compose.git
synced 2025-11-09 16:29:58 +01:00
135 lines
4.0 KiB
Go
135 lines
4.0 KiB
Go
/*
|
|
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
|
|
}
|