-
Notifications
You must be signed in to change notification settings - Fork 4.7k
/
Copy pathcopyimage.go
98 lines (79 loc) · 2.87 KB
/
copyimage.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*
Copyright 2021 The Kubernetes 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 assets
import (
"fmt"
"github.com/google/go-containerregistry/pkg/authn"
"github.com/google/go-containerregistry/pkg/name"
"github.com/google/go-containerregistry/pkg/v1/remote"
"github.com/google/go-containerregistry/pkg/v1/types"
"k8s.io/klog/v2"
)
// CopyImage copies a docker image from a source registry, to a target registry,
// typically used for highly secure clusters.
type CopyImage struct {
Name string
SourceImage string
TargetImage string
}
func (e *CopyImage) Run() error {
source := e.SourceImage
target := e.TargetImage
sourceRef, err := name.ParseReference(source)
if err != nil {
return fmt.Errorf("parsing reference %q: %v", source, err)
}
targetRef, err := name.ParseReference(target)
if err != nil {
return fmt.Errorf("parsing reference for %q: %v", target, err)
}
options := []remote.Option{remote.WithAuthFromKeychain(authn.DefaultKeychain)}
desc, err := remote.Get(sourceRef, options...)
if err != nil {
return fmt.Errorf("fetching %q: %v", source, err)
}
targetDesc, err := remote.Get(targetRef, options...)
if err == nil && desc.Digest.String() == targetDesc.Digest.String() {
klog.Infof("no need to copy image from %v to %v", sourceRef, targetRef)
return nil
}
switch desc.MediaType {
case types.OCIImageIndex, types.DockerManifestList:
// Handle indexes separately.
if err := copyIndex(desc, sourceRef, targetRef, options...); err != nil {
return fmt.Errorf("failed to copy index: %v", err)
}
default:
// Assume anything else is an image, since some registries don't set mediaTypes properly.
if err := copyImage(desc, sourceRef, targetRef, options...); err != nil {
return fmt.Errorf("failed to copy image: %v", err)
}
}
return nil
}
func copyImage(desc *remote.Descriptor, sourceRef name.Reference, targetRef name.Reference, options ...remote.Option) error {
klog.Infof("copying image from %v to %v", sourceRef, targetRef)
img, err := desc.Image()
if err != nil {
return err
}
return remote.Write(targetRef, img, options...)
}
func copyIndex(desc *remote.Descriptor, sourceRef name.Reference, targetRef name.Reference, options ...remote.Option) error {
klog.Infof("copying image index from %v to %v", sourceRef, targetRef)
idx, err := desc.ImageIndex()
if err != nil {
return err
}
return remote.WriteIndex(targetRef, idx, options...)
}