image_pull.go 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  1. package containerd
  2. import (
  3. "context"
  4. "errors"
  5. "io"
  6. "github.com/containerd/containerd"
  7. "github.com/containerd/containerd/images"
  8. "github.com/containerd/containerd/platforms"
  9. "github.com/docker/distribution"
  10. "github.com/docker/distribution/reference"
  11. "github.com/docker/docker/api/types/registry"
  12. "github.com/docker/docker/errdefs"
  13. "github.com/docker/docker/pkg/streamformatter"
  14. "github.com/opencontainers/go-digest"
  15. specs "github.com/opencontainers/image-spec/specs-go/v1"
  16. )
  17. // PullImage initiates a pull operation. image is the repository name to pull, and
  18. // tagOrDigest may be either empty, or indicate a specific tag or digest to pull.
  19. func (i *ImageService) PullImage(ctx context.Context, image, tagOrDigest string, platform *specs.Platform, metaHeaders map[string][]string, authConfig *registry.AuthConfig, outStream io.Writer) error {
  20. var opts []containerd.RemoteOpt
  21. if platform != nil {
  22. opts = append(opts, containerd.WithPlatform(platforms.Format(*platform)))
  23. }
  24. ref, err := reference.ParseNormalizedNamed(image)
  25. if err != nil {
  26. return errdefs.InvalidParameter(err)
  27. }
  28. // TODO(thaJeztah) this could use a WithTagOrDigest() utility
  29. if tagOrDigest != "" {
  30. // The "tag" could actually be a digest.
  31. var dgst digest.Digest
  32. dgst, err = digest.Parse(tagOrDigest)
  33. if err == nil {
  34. ref, err = reference.WithDigest(reference.TrimNamed(ref), dgst)
  35. } else {
  36. ref, err = reference.WithTag(ref, tagOrDigest)
  37. }
  38. if err != nil {
  39. return errdefs.InvalidParameter(err)
  40. }
  41. }
  42. resolver, _ := i.newResolverFromAuthConfig(authConfig)
  43. opts = append(opts, containerd.WithResolver(resolver))
  44. jobs := newJobs()
  45. h := images.HandlerFunc(func(ctx context.Context, desc specs.Descriptor) ([]specs.Descriptor, error) {
  46. if desc.MediaType != images.MediaTypeDockerSchema1Manifest {
  47. jobs.Add(desc)
  48. }
  49. return nil, nil
  50. })
  51. opts = append(opts, containerd.WithImageHandler(h))
  52. out := streamformatter.NewJSONProgressOutput(outStream, false)
  53. finishProgress := jobs.showProgress(ctx, out, pullProgress{Store: i.client.ContentStore(), ShowExists: true})
  54. defer finishProgress()
  55. opts = append(opts, containerd.WithPullUnpack)
  56. opts = append(opts, containerd.WithPullSnapshotter(i.snapshotter))
  57. _, err = i.client.Pull(ctx, ref.String(), opts...)
  58. return err
  59. }
  60. // GetRepository returns a repository from the registry.
  61. func (i *ImageService) GetRepository(ctx context.Context, ref reference.Named, authConfig *registry.AuthConfig) (distribution.Repository, error) {
  62. return nil, errdefs.NotImplemented(errors.New("not implemented"))
  63. }