Add --all-tags to pull command
Signed-off-by: TomSweeneyRedHat <tsweeney@redhat.com> Add --all-tags for the `podman pull` command so all tags of an image will be pulled, not just ':latest'. Emulates the change in Buildah https://github.com/containers/buildah/pull/1263 Signed-off-by: TomSweeneyRedHat <tsweeney@redhat.com>
This commit is contained in:
		
							parent
							
								
									c86e8f180c
								
							
						
					
					
						commit
						1543a01be8
					
				|  | @ -334,6 +334,7 @@ type PsValues struct { | ||||||
| 
 | 
 | ||||||
| type PullValues struct { | type PullValues struct { | ||||||
| 	PodmanCommand | 	PodmanCommand | ||||||
|  | 	AllTags         bool | ||||||
| 	Authfile        string | 	Authfile        string | ||||||
| 	CertDir         string | 	CertDir         string | ||||||
| 	Creds           string | 	Creds           string | ||||||
|  |  | ||||||
|  | @ -6,11 +6,13 @@ import ( | ||||||
| 	"os" | 	"os" | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/containers/image/docker" | ||||||
| 	dockerarchive "github.com/containers/image/docker/archive" | 	dockerarchive "github.com/containers/image/docker/archive" | ||||||
| 	"github.com/containers/image/transports/alltransports" | 	"github.com/containers/image/transports/alltransports" | ||||||
| 	"github.com/containers/image/types" | 	"github.com/containers/image/types" | ||||||
| 	"github.com/containers/libpod/cmd/podman/cliconfig" | 	"github.com/containers/libpod/cmd/podman/cliconfig" | ||||||
| 	"github.com/containers/libpod/libpod/adapter" | 	"github.com/containers/libpod/libpod/adapter" | ||||||
|  | 	"github.com/containers/libpod/libpod/common" | ||||||
| 	image2 "github.com/containers/libpod/libpod/image" | 	image2 "github.com/containers/libpod/libpod/image" | ||||||
| 	"github.com/containers/libpod/pkg/util" | 	"github.com/containers/libpod/pkg/util" | ||||||
| 	"github.com/pkg/errors" | 	"github.com/pkg/errors" | ||||||
|  | @ -41,6 +43,7 @@ specified, the image with the 'latest' tag (if it exists) is pulled | ||||||
| func init() { | func init() { | ||||||
| 	pullCommand.Command = _pullCommand | 	pullCommand.Command = _pullCommand | ||||||
| 	flags := pullCommand.Flags() | 	flags := pullCommand.Flags() | ||||||
|  | 	flags.BoolVar(&pullCommand.AllTags, "all-tags", false, "All tagged images inthe repository will be pulled") | ||||||
| 	flags.StringVar(&pullCommand.Authfile, "authfile", "", "Path of the authentication file. Default is ${XDG_RUNTIME_DIR}/containers/auth.json. Use REGISTRY_AUTH_FILE environment variable to override") | 	flags.StringVar(&pullCommand.Authfile, "authfile", "", "Path of the authentication file. Default is ${XDG_RUNTIME_DIR}/containers/auth.json. Use REGISTRY_AUTH_FILE environment variable to override") | ||||||
| 	flags.StringVar(&pullCommand.CertDir, "cert-dir", "", "`Pathname` of a directory containing TLS certificates and keys") | 	flags.StringVar(&pullCommand.CertDir, "cert-dir", "", "`Pathname` of a directory containing TLS certificates and keys") | ||||||
| 	flags.StringVar(&pullCommand.Creds, "creds", "", "`Credentials` (USERNAME:PASSWORD) to use for authenticating to a registry") | 	flags.StringVar(&pullCommand.Creds, "creds", "", "`Credentials` (USERNAME:PASSWORD) to use for authenticating to a registry") | ||||||
|  | @ -69,6 +72,15 @@ func pullCmd(c *cliconfig.PullValues) error { | ||||||
| 		logrus.Errorf("too many arguments. Requires exactly 1") | 		logrus.Errorf("too many arguments. Requires exactly 1") | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	arr := strings.SplitN(args[0], ":", 2) | ||||||
|  | 	if len(arr) == 2 { | ||||||
|  | 		if c.Bool("all-tags") { | ||||||
|  | 			logrus.Errorf("tag can't be used with --all-tags") | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	ctx := getContext() | ||||||
| 	image := args[0] | 	image := args[0] | ||||||
| 
 | 
 | ||||||
| 	var registryCreds *types.DockerAuthConfig | 	var registryCreds *types.DockerAuthConfig | ||||||
|  | @ -83,7 +95,6 @@ func pullCmd(c *cliconfig.PullValues) error { | ||||||
| 
 | 
 | ||||||
| 	var ( | 	var ( | ||||||
| 		writer io.Writer | 		writer io.Writer | ||||||
| 		imgID  string |  | ||||||
| 	) | 	) | ||||||
| 	if !c.Quiet { | 	if !c.Quiet { | ||||||
| 		writer = os.Stderr | 		writer = os.Stderr | ||||||
|  | @ -107,18 +118,58 @@ func pullCmd(c *cliconfig.PullValues) error { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return errors.Wrapf(err, "error pulling image from %q", image) | 			return errors.Wrapf(err, "error pulling image from %q", image) | ||||||
| 		} | 		} | ||||||
| 		imgID = newImage[0].ID() | 		fmt.Println(newImage[0].ID()) | ||||||
| 	} else { | 	} else { | ||||||
| 		authfile := getAuthFile(c.Authfile) | 		authfile := getAuthFile(c.String("authfile")) | ||||||
| 		newImage, err := runtime.New(getContext(), image, c.SignaturePolicy, authfile, writer, &dockerRegistryOptions, image2.SigningOptions{}, true, nil) | 		spec := image | ||||||
|  | 		systemContext := common.GetSystemContext("", authfile, false) | ||||||
|  | 		srcRef, err := alltransports.ParseImageName(spec) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
|  | 			dockerTransport := "docker://" | ||||||
|  | 			logrus.Debugf("error parsing image name %q, trying with transport %q: %v", spec, dockerTransport, err) | ||||||
|  | 			spec = dockerTransport + spec | ||||||
|  | 			srcRef2, err2 := alltransports.ParseImageName(spec) | ||||||
|  | 			if err2 != nil { | ||||||
|  | 				return errors.Wrapf(err2, "error parsing image name %q", image) | ||||||
|  | 			} | ||||||
|  | 			srcRef = srcRef2 | ||||||
|  | 		} | ||||||
|  | 		var names []string | ||||||
|  | 		if c.Bool("all-tags") { | ||||||
|  | 			if srcRef.DockerReference() == nil { | ||||||
|  | 				return errors.New("Non-docker transport is currently not supported") | ||||||
|  | 			} | ||||||
|  | 			tags, err := docker.GetRepositoryTags(ctx, systemContext, srcRef) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return errors.Wrapf(err, "error getting repository tags") | ||||||
|  | 			} | ||||||
|  | 			for _, tag := range tags { | ||||||
|  | 				name := spec + ":" + tag | ||||||
|  | 				names = append(names, name) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			names = append(names, spec) | ||||||
|  | 		} | ||||||
|  | 		var foundIDs []string | ||||||
|  | 		foundImage := true | ||||||
|  | 		for _, name := range names { | ||||||
|  | 			newImage, err := runtime.New(getContext(), name, c.String("signature-policy"), authfile, writer, &dockerRegistryOptions, image2.SigningOptions{}, true, nil) | ||||||
|  | 			if err != nil { | ||||||
|  | 				println(errors.Wrapf(err, "error pulling image %q", name)) | ||||||
|  | 				foundImage = false | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			foundIDs = append(foundIDs, newImage.ID()) | ||||||
|  | 		} | ||||||
|  | 		if len(names) == 1 && !foundImage { | ||||||
| 			return errors.Wrapf(err, "error pulling image %q", image) | 			return errors.Wrapf(err, "error pulling image %q", image) | ||||||
| 		} | 		} | ||||||
| 		imgID = newImage.ID() | 		if len(names) > 1 { | ||||||
| 	} | 			fmt.Println("Pulled Images:") | ||||||
| 
 | 		} | ||||||
| 	// Intentionally choosing to ignore if there is an error because
 | 		for _, id := range foundIDs { | ||||||
| 	// outputting the image ID is a NTH and not integral to the pull
 | 			fmt.Println(id) | ||||||
| 	fmt.Println(imgID) | 		} | ||||||
|  | 	} // end else if strings.HasPrefix(image, dockerarchive.Transport.Name()+":")
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -45,6 +45,10 @@ Image stored in local container/storage | ||||||
| 
 | 
 | ||||||
| ## OPTIONS | ## OPTIONS | ||||||
| 
 | 
 | ||||||
|  | **--all-tags, a** | ||||||
|  | 
 | ||||||
|  | All tagged images in the repository will be pulled. | ||||||
|  | 
 | ||||||
| **--authfile** | **--authfile** | ||||||
| 
 | 
 | ||||||
| Path of the authentication file. Default is ${XDG_RUNTIME\_DIR}/containers/auth.json, which is set using `podman login`. | Path of the authentication file. Default is ${XDG_RUNTIME\_DIR}/containers/auth.json, which is set using `podman login`. | ||||||
|  |  | ||||||
|  | @ -163,4 +163,20 @@ var _ = Describe("Podman pull", func() { | ||||||
| 
 | 
 | ||||||
| 		Expect(pull.OutputToString()).To(ContainSubstring(shortImageId)) | 		Expect(pull.OutputToString()).To(ContainSubstring(shortImageId)) | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
|  | 	It("podman pull check all tags", func() { | ||||||
|  | 		session := podmanTest.Podman([]string{"pull", "--all-tags", "alpine"}) | ||||||
|  | 		session.WaitWithDefaultTimeout() | ||||||
|  | 		Expect(session.ExitCode()).To(Equal(0)) | ||||||
|  | 		Expect(session.LineInOuputStartsWith("Pulled Images:")).To(BeTrue()) | ||||||
|  | 
 | ||||||
|  | 		session = podmanTest.Podman([]string{"images"}) | ||||||
|  | 		session.WaitWithDefaultTimeout() | ||||||
|  | 		Expect(session.ExitCode()).To(Equal(0)) | ||||||
|  | 		Expect(len(session.OutputToStringArray())).To(BeNumerically(">", 4)) | ||||||
|  | 
 | ||||||
|  | 		rmi := podmanTest.Podman([]string{"rmi", "-a", "-f"}) | ||||||
|  | 		rmi.WaitWithDefaultTimeout() | ||||||
|  | 		Expect(rmi.ExitCode()).To(Equal(0)) | ||||||
|  | 	}) | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | @ -270,7 +270,7 @@ func (s *PodmanSession) LineInOuputStartsWith(term string) bool { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| //LineInOutputContains returns true if a line in a
 | //LineInOutputContains returns true if a line in a
 | ||||||
| // session output starts with the supplied string
 | // session output contains the supplied string
 | ||||||
| func (s *PodmanSession) LineInOutputContains(term string) bool { | func (s *PodmanSession) LineInOutputContains(term string) bool { | ||||||
| 	for _, i := range s.OutputToStringArray() { | 	for _, i := range s.OutputToStringArray() { | ||||||
| 		if strings.Contains(i, term) { | 		if strings.Contains(i, term) { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue