@@ -20,10 +20,11 @@ import (
20
20
"fmt"
21
21
"io"
22
22
"os"
23
+ "time"
23
24
25
+ "github.com/containerd/containerd"
24
26
"github.com/containerd/containerd/cmd/ctr/commands"
25
- "github.com/containerd/containerd/images"
26
- oci "github.com/containerd/containerd/images/oci"
27
+ "github.com/containerd/containerd/images/archive"
27
28
"github.com/containerd/containerd/log"
28
29
"github.com/urfave/cli"
29
30
)
@@ -34,45 +35,58 @@ var importCommand = cli.Command{
34
35
ArgsUsage : "[flags] <in>" ,
35
36
Description : `Import images from a tar stream.
36
37
Implemented formats:
37
- - oci.v1 (default)
38
+ - oci.v1
39
+ - docker.v1.1
40
+ - docker.v1.2
38
41
39
42
40
- For oci.v1 format, you need to specify --oci-name because an OCI archive contains image refs (tags)
41
- but does not contain the base image name.
43
+ For OCI v1, you may need to specify --base-name because an OCI archive may
44
+ contain only partial image references (tags without the base image name).
45
+ If no base image name is provided, a name will be generated as "import-%{yyyy-MM-dd}".
42
46
43
47
e.g.
44
- $ ctr images import --format oci.v1 --oci -name foo/bar foobar.tar
48
+ $ ctr images import --base -name foo/bar foobar.tar
45
49
46
50
If foobar.tar contains an OCI ref named "latest" and anonymous ref "sha256:deadbeef", the command will create
47
51
"foo/bar:latest" and "foo/bar@sha256:deadbeef" images in the containerd store.
48
52
` ,
49
53
Flags : append ([]cli.Flag {
50
54
cli.StringFlag {
51
- Name : "format" ,
52
- Value : "oci.v1" ,
53
- Usage : "image format. See DESCRIPTION." ,
55
+ Name : "base-name" ,
56
+ Value : "" ,
57
+ Usage : "base image name for added images, when provided only images with this name prefix are imported" ,
58
+ },
59
+ cli.BoolFlag {
60
+ Name : "digests" ,
61
+ Usage : "whether to create digest images (default: false)" ,
54
62
},
55
63
cli.StringFlag {
56
- Name : "oci-name" ,
57
- Value : "unknown/unknown" ,
58
- Usage : "prefix added to either oci.v1 ref annotation or digest" ,
64
+ Name : "index-name" ,
65
+ Usage : "image name to keep index as, by default index is discarded" ,
59
66
},
60
- // TODO(AkihiroSuda): support commands.LabelFlag (for all children objects)
61
67
}, commands .SnapshotterFlags ... ),
62
68
63
69
Action : func (context * cli.Context ) error {
64
70
var (
65
- in = context .Args ().First ()
66
- imageImporter images. Importer
71
+ in = context .Args ().First ()
72
+ opts []containerd. ImportOpt
67
73
)
68
74
69
- switch format := context .String ("format" ); format {
70
- case "oci.v1" :
71
- imageImporter = & oci.V1Importer {
72
- ImageName : context .String ("oci-name" ),
73
- }
74
- default :
75
- return fmt .Errorf ("unknown format %s" , format )
75
+ prefix := context .String ("base-name" )
76
+ if prefix == "" {
77
+ prefix = fmt .Sprintf ("import-%s" , time .Now ().Format ("2006-01-02" ))
78
+ opts = append (opts , containerd .WithImageRefTranslator (archive .AddRefPrefix (prefix )))
79
+ } else {
80
+ // When provided, filter out references which do not match
81
+ opts = append (opts , containerd .WithImageRefTranslator (archive .FilterRefPrefix (prefix )))
82
+ }
83
+
84
+ if context .Bool ("digests" ) {
85
+ opts = append (opts , containerd .WithDigestRef (archive .DigestTranslator (prefix )))
86
+ }
87
+
88
+ if idxName := context .String ("index-name" ); idxName != "" {
89
+ opts = append (opts , containerd .WithIndexName (idxName ))
76
90
}
77
91
78
92
client , ctx , cancel , err := commands .NewClient (context )
@@ -90,20 +104,24 @@ If foobar.tar contains an OCI ref named "latest" and anonymous ref "sha256:deadb
90
104
return err
91
105
}
92
106
}
93
- imgs , err := client .Import (ctx , imageImporter , r )
107
+ imgs , err := client .Import (ctx , r , opts ... )
108
+ closeErr := r .Close ()
94
109
if err != nil {
95
110
return err
96
111
}
97
- if err = r . Close (); err != nil {
98
- return err
112
+ if closeErr != nil {
113
+ return closeErr
99
114
}
100
115
101
116
log .G (ctx ).Debugf ("unpacking %d images" , len (imgs ))
102
117
103
118
for _ , img := range imgs {
119
+ // TODO: Allow configuration of the platform
120
+ image := containerd .NewImage (client , img )
121
+
104
122
// TODO: Show unpack status
105
- fmt .Printf ("unpacking %s (%s)..." , img .Name () , img .Target () .Digest )
106
- err = img .Unpack (ctx , context .String ("snapshotter" ))
123
+ fmt .Printf ("unpacking %s (%s)..." , img .Name , img .Target .Digest )
124
+ err = image .Unpack (ctx , context .String ("snapshotter" ))
107
125
if err != nil {
108
126
return err
109
127
}
0 commit comments