@@ -45,6 +45,7 @@ import (
45
45
"github.com/containerd/containerd/images"
46
46
"github.com/containerd/containerd/namespaces"
47
47
"github.com/containerd/containerd/pkg/dialer"
48
+ "github.com/containerd/containerd/platforms"
48
49
"github.com/containerd/containerd/plugin"
49
50
"github.com/containerd/containerd/remotes"
50
51
"github.com/containerd/containerd/remotes/docker"
@@ -284,98 +285,137 @@ func defaultRemoteContext() *RemoteContext {
284
285
}
285
286
}
286
287
288
+ // Fetch downloads the provided content into containerd's content store
289
+ // and returns a non-platform specific image reference
290
+ func (c * Client ) Fetch (ctx context.Context , ref string , opts ... RemoteOpt ) (images.Image , error ) {
291
+ fetchCtx := defaultRemoteContext ()
292
+ for _ , o := range opts {
293
+ if err := o (c , fetchCtx ); err != nil {
294
+ return images.Image {}, err
295
+ }
296
+ }
297
+
298
+ if fetchCtx .Unpack {
299
+ return images.Image {}, errors .New ("unpack on fetch not supported, try pull" )
300
+ }
301
+
302
+ ctx , done , err := c .WithLease (ctx )
303
+ if err != nil {
304
+ return images.Image {}, err
305
+ }
306
+ defer done (ctx )
307
+
308
+ return c .fetch (ctx , fetchCtx , ref )
309
+ }
310
+
287
311
// Pull downloads the provided content into containerd's content store
312
+ // and returns a platform specific image object
288
313
func (c * Client ) Pull (ctx context.Context , ref string , opts ... RemoteOpt ) (Image , error ) {
289
314
pullCtx := defaultRemoteContext ()
290
315
for _ , o := range opts {
291
316
if err := o (c , pullCtx ); err != nil {
292
317
return nil , err
293
318
}
294
319
}
295
- store := c .ContentStore ()
320
+
321
+ if len (pullCtx .Platforms ) > 1 {
322
+ return nil , errors .New ("cannot pull multiplatform image locally, try Fetch" )
323
+ } else if len (pullCtx .Platforms ) == 0 {
324
+ pullCtx .Platforms = []string {platforms .Default ()}
325
+ }
296
326
297
327
ctx , done , err := c .WithLease (ctx )
298
328
if err != nil {
299
329
return nil , err
300
330
}
301
331
defer done (ctx )
302
332
303
- name , desc , err := pullCtx . Resolver . Resolve (ctx , ref )
333
+ img , err := c . fetch (ctx , pullCtx , ref )
304
334
if err != nil {
305
- return nil , errors . Wrapf ( err , "failed to resolve reference %q" , ref )
335
+ return nil , err
306
336
}
307
337
308
- fetcher , err := pullCtx .Resolver .Fetcher (ctx , name )
338
+ i := NewImageWithPlatform (c , img , pullCtx .Platforms [0 ])
339
+
340
+ if pullCtx .Unpack {
341
+ if err := i .Unpack (ctx , pullCtx .Snapshotter ); err != nil {
342
+ return nil , errors .Wrapf (err , "failed to unpack image on snapshotter %s" , pullCtx .Snapshotter )
343
+ }
344
+ }
345
+
346
+ return i , nil
347
+ }
348
+
349
+ func (c * Client ) fetch (ctx context.Context , rCtx * RemoteContext , ref string ) (images.Image , error ) {
350
+ store := c .ContentStore ()
351
+ name , desc , err := rCtx .Resolver .Resolve (ctx , ref )
309
352
if err != nil {
310
- return nil , errors .Wrapf (err , "failed to get fetcher for %q" , name )
353
+ return images.Image {}, errors .Wrapf (err , "failed to resolve reference %q" , ref )
354
+ }
355
+
356
+ fetcher , err := rCtx .Resolver .Fetcher (ctx , name )
357
+ if err != nil {
358
+ return images.Image {}, errors .Wrapf (err , "failed to get fetcher for %q" , name )
311
359
}
312
360
313
361
var (
314
362
schema1Converter * schema1.Converter
315
363
handler images.Handler
316
364
)
317
- if desc .MediaType == images .MediaTypeDockerSchema1Manifest && pullCtx .ConvertSchema1 {
365
+ if desc .MediaType == images .MediaTypeDockerSchema1Manifest && rCtx .ConvertSchema1 {
318
366
schema1Converter = schema1 .NewConverter (store , fetcher )
319
- handler = images .Handlers (append (pullCtx .BaseHandlers , schema1Converter )... )
367
+ handler = images .Handlers (append (rCtx .BaseHandlers , schema1Converter )... )
320
368
} else {
321
369
// Get all the children for a descriptor
322
370
childrenHandler := images .ChildrenHandler (store )
323
371
// Set any children labels for that content
324
372
childrenHandler = images .SetChildrenLabels (store , childrenHandler )
325
373
// Filter children by platforms
326
- childrenHandler = images .FilterPlatforms (childrenHandler , pullCtx .Platforms ... )
374
+ childrenHandler = images .FilterPlatforms (childrenHandler , rCtx .Platforms ... )
327
375
328
- handler = images .Handlers (append (pullCtx .BaseHandlers ,
376
+ handler = images .Handlers (append (rCtx .BaseHandlers ,
329
377
remotes .FetchHandler (store , fetcher ),
330
378
childrenHandler ,
331
379
)... )
332
380
}
333
381
334
382
if err := images .Dispatch (ctx , handler , desc ); err != nil {
335
- return nil , err
383
+ return images. Image {} , err
336
384
}
337
385
if schema1Converter != nil {
338
386
desc , err = schema1Converter .Convert (ctx )
339
387
if err != nil {
340
- return nil , err
388
+ return images. Image {} , err
341
389
}
342
390
}
343
391
344
- img := & image {
345
- client : c ,
346
- i : images.Image {
347
- Name : name ,
348
- Target : desc ,
349
- Labels : pullCtx .Labels ,
350
- },
351
- }
352
-
353
- if pullCtx .Unpack {
354
- if err := img .Unpack (ctx , pullCtx .Snapshotter ); err != nil {
355
- return nil , errors .Wrapf (err , "failed to unpack image on snapshotter %s" , pullCtx .Snapshotter )
356
- }
392
+ img := images.Image {
393
+ Name : name ,
394
+ Target : desc ,
395
+ Labels : rCtx .Labels ,
357
396
}
358
397
359
398
is := c .ImageService ()
360
399
for {
361
- if created , err := is .Create (ctx , img . i ); err != nil {
400
+ if created , err := is .Create (ctx , img ); err != nil {
362
401
if ! errdefs .IsAlreadyExists (err ) {
363
- return nil , err
402
+ return images. Image {} , err
364
403
}
365
404
366
- updated , err := is .Update (ctx , img . i )
405
+ updated , err := is .Update (ctx , img )
367
406
if err != nil {
368
407
// if image was removed, try create again
369
408
if errdefs .IsNotFound (err ) {
370
409
continue
371
410
}
372
- return nil , err
411
+ return images. Image {} , err
373
412
}
374
413
375
- img . i = updated
414
+ img = updated
376
415
} else {
377
- img . i = created
416
+ img = created
378
417
}
418
+
379
419
return img , nil
380
420
}
381
421
}
0 commit comments