mirror of
				https://github.com/containers/podman.git
				synced 2025-10-31 18:08:51 +08:00 
			
		
		
		
	migrate Podman to containers/common/libimage
Migrate the Podman code base over to `common/libimage` which replaces `libpod/image` and a lot of glue code entirely. Note that I tried to leave bread crumbs for changed tests. Miscellaneous changes: * Some errors yield different messages which required to alter some tests. * I fixed some pre-existing issues in the code. Others were marked as `//TODO`s to prevent the PR from exploding. * The `NamesHistory` of an image is returned as is from the storage. Previously, we did some filtering which I think is undesirable. Instead we should return the data as stored in the storage. * Touched handlers use the ABI interfaces where possible. * Local image resolution: previously Podman would match "foo" on "myfoo". This behaviour has been changed and Podman will now only match on repository boundaries such that "foo" would match "my/foo" but not "myfoo". I consider the old behaviour to be a bug, at the very least an exotic corner case. * Futhermore, "foo:none" does *not* resolve to a local image "foo" without tag anymore. It's a hill I am (almost) willing to die on. * `image prune` prints the IDs of pruned images. Previously, in some cases, the names were printed instead. The API clearly states ID, so we should stick to it. * Compat endpoint image removal with _force_ deletes the entire not only the specified tag. Signed-off-by: Valentin Rothberg <rothberg@redhat.com>
This commit is contained in:
		
							
								
								
									
										242
									
								
								vendor/github.com/containers/common/libimage/image_config.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								vendor/github.com/containers/common/libimage/image_config.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,242 @@ | ||||
| package libimage | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/containers/common/pkg/signal" | ||||
| 	ociv1 "github.com/opencontainers/image-spec/specs-go/v1" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
|  | ||||
| // ImageConfig is a wrapper around the OCIv1 Image Configuration struct exported | ||||
| // by containers/image, but containing additional fields that are not supported | ||||
| // by OCIv1 (but are by Docker v2) - notably OnBuild. | ||||
| type ImageConfig struct { | ||||
| 	ociv1.ImageConfig | ||||
| 	OnBuild []string | ||||
| } | ||||
|  | ||||
| // ImageConfigFromChanges produces a v1.ImageConfig from the --change flag that | ||||
| // is accepted by several Podman commands. It accepts a (limited subset) of | ||||
| // Dockerfile instructions. | ||||
| // Valid changes are: | ||||
| // * USER | ||||
| // * EXPOSE | ||||
| // * ENV | ||||
| // * ENTRYPOINT | ||||
| // * CMD | ||||
| // * VOLUME | ||||
| // * WORKDIR | ||||
| // * LABEL | ||||
| // * STOPSIGNAL | ||||
| // * ONBUILD | ||||
| func ImageConfigFromChanges(changes []string) (*ImageConfig, error) { // nolint:gocyclo | ||||
| 	config := &ImageConfig{} | ||||
|  | ||||
| 	for _, change := range changes { | ||||
| 		// First, let's assume proper Dockerfile format - space | ||||
| 		// separator between instruction and value | ||||
| 		split := strings.SplitN(change, " ", 2) | ||||
|  | ||||
| 		if len(split) != 2 { | ||||
| 			split = strings.SplitN(change, "=", 2) | ||||
| 			if len(split) != 2 { | ||||
| 				return nil, errors.Errorf("invalid change %q - must be formatted as KEY VALUE", change) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		outerKey := strings.ToUpper(strings.TrimSpace(split[0])) | ||||
| 		value := strings.TrimSpace(split[1]) | ||||
| 		switch outerKey { | ||||
| 		case "USER": | ||||
| 			// Assume literal contents are the user. | ||||
| 			if value == "" { | ||||
| 				return nil, errors.Errorf("invalid change %q - must provide a value to USER", change) | ||||
| 			} | ||||
| 			config.User = value | ||||
| 		case "EXPOSE": | ||||
| 			// EXPOSE is either [portnum] or | ||||
| 			// [portnum]/[proto] | ||||
| 			// Protocol must be "tcp" or "udp" | ||||
| 			splitPort := strings.Split(value, "/") | ||||
| 			if len(splitPort) > 2 { | ||||
| 				return nil, errors.Errorf("invalid change %q - EXPOSE port must be formatted as PORT[/PROTO]", change) | ||||
| 			} | ||||
| 			portNum, err := strconv.Atoi(splitPort[0]) | ||||
| 			if err != nil { | ||||
| 				return nil, errors.Wrapf(err, "invalid change %q - EXPOSE port must be an integer", change) | ||||
| 			} | ||||
| 			if portNum > 65535 || portNum <= 0 { | ||||
| 				return nil, errors.Errorf("invalid change %q - EXPOSE port must be a valid port number", change) | ||||
| 			} | ||||
| 			proto := "tcp" | ||||
| 			if len(splitPort) > 1 { | ||||
| 				testProto := strings.ToLower(splitPort[1]) | ||||
| 				switch testProto { | ||||
| 				case "tcp", "udp": | ||||
| 					proto = testProto | ||||
| 				default: | ||||
| 					return nil, errors.Errorf("invalid change %q - EXPOSE protocol must be TCP or UDP", change) | ||||
| 				} | ||||
| 			} | ||||
| 			if config.ExposedPorts == nil { | ||||
| 				config.ExposedPorts = make(map[string]struct{}) | ||||
| 			} | ||||
| 			config.ExposedPorts[fmt.Sprintf("%d/%s", portNum, proto)] = struct{}{} | ||||
| 		case "ENV": | ||||
| 			// Format is either: | ||||
| 			// ENV key=value | ||||
| 			// ENV key=value key=value ... | ||||
| 			// ENV key value | ||||
| 			// Both keys and values can be surrounded by quotes to group them. | ||||
| 			// For now: we only support key=value | ||||
| 			// We will attempt to strip quotation marks if present. | ||||
|  | ||||
| 			var ( | ||||
| 				key, val string | ||||
| 			) | ||||
|  | ||||
| 			splitEnv := strings.SplitN(value, "=", 2) | ||||
| 			key = splitEnv[0] | ||||
| 			// We do need a key | ||||
| 			if key == "" { | ||||
| 				return nil, errors.Errorf("invalid change %q - ENV must have at least one argument", change) | ||||
| 			} | ||||
| 			// Perfectly valid to not have a value | ||||
| 			if len(splitEnv) == 2 { | ||||
| 				val = splitEnv[1] | ||||
| 			} | ||||
|  | ||||
| 			if strings.HasPrefix(key, `"`) && strings.HasSuffix(key, `"`) { | ||||
| 				key = strings.TrimPrefix(strings.TrimSuffix(key, `"`), `"`) | ||||
| 			} | ||||
| 			if strings.HasPrefix(val, `"`) && strings.HasSuffix(val, `"`) { | ||||
| 				val = strings.TrimPrefix(strings.TrimSuffix(val, `"`), `"`) | ||||
| 			} | ||||
| 			config.Env = append(config.Env, fmt.Sprintf("%s=%s", key, val)) | ||||
| 		case "ENTRYPOINT": | ||||
| 			// Two valid forms. | ||||
| 			// First, JSON array. | ||||
| 			// Second, not a JSON array - we interpret this as an | ||||
| 			// argument to `sh -c`, unless empty, in which case we | ||||
| 			// just use a blank entrypoint. | ||||
| 			testUnmarshal := []string{} | ||||
| 			if err := json.Unmarshal([]byte(value), &testUnmarshal); err != nil { | ||||
| 				// It ain't valid JSON, so assume it's an | ||||
| 				// argument to sh -c if not empty. | ||||
| 				if value != "" { | ||||
| 					config.Entrypoint = []string{"/bin/sh", "-c", value} | ||||
| 				} else { | ||||
| 					config.Entrypoint = []string{} | ||||
| 				} | ||||
| 			} else { | ||||
| 				// Valid JSON | ||||
| 				config.Entrypoint = testUnmarshal | ||||
| 			} | ||||
| 		case "CMD": | ||||
| 			// Same valid forms as entrypoint. | ||||
| 			// However, where ENTRYPOINT assumes that 'ENTRYPOINT ' | ||||
| 			// means no entrypoint, CMD assumes it is 'sh -c' with | ||||
| 			// no third argument. | ||||
| 			testUnmarshal := []string{} | ||||
| 			if err := json.Unmarshal([]byte(value), &testUnmarshal); err != nil { | ||||
| 				// It ain't valid JSON, so assume it's an | ||||
| 				// argument to sh -c. | ||||
| 				// Only include volume if it's not "" | ||||
| 				config.Cmd = []string{"/bin/sh", "-c"} | ||||
| 				if value != "" { | ||||
| 					config.Cmd = append(config.Cmd, value) | ||||
| 				} | ||||
| 			} else { | ||||
| 				// Valid JSON | ||||
| 				config.Cmd = testUnmarshal | ||||
| 			} | ||||
| 		case "VOLUME": | ||||
| 			// Either a JSON array or a set of space-separated | ||||
| 			// paths. | ||||
| 			// Acts rather similar to ENTRYPOINT and CMD, but always | ||||
| 			// appends rather than replacing, and no sh -c prepend. | ||||
| 			testUnmarshal := []string{} | ||||
| 			if err := json.Unmarshal([]byte(value), &testUnmarshal); err != nil { | ||||
| 				// Not valid JSON, so split on spaces | ||||
| 				testUnmarshal = strings.Split(value, " ") | ||||
| 			} | ||||
| 			if len(testUnmarshal) == 0 { | ||||
| 				return nil, errors.Errorf("invalid change %q - must provide at least one argument to VOLUME", change) | ||||
| 			} | ||||
| 			for _, vol := range testUnmarshal { | ||||
| 				if vol == "" { | ||||
| 					return nil, errors.Errorf("invalid change %q - VOLUME paths must not be empty", change) | ||||
| 				} | ||||
| 				if config.Volumes == nil { | ||||
| 					config.Volumes = make(map[string]struct{}) | ||||
| 				} | ||||
| 				config.Volumes[vol] = struct{}{} | ||||
| 			} | ||||
| 		case "WORKDIR": | ||||
| 			// This can be passed multiple times. | ||||
| 			// Each successive invocation is treated as relative to | ||||
| 			// the previous one - so WORKDIR /A, WORKDIR b, | ||||
| 			// WORKDIR c results in /A/b/c | ||||
| 			// Just need to check it's not empty... | ||||
| 			if value == "" { | ||||
| 				return nil, errors.Errorf("invalid change %q - must provide a non-empty WORKDIR", change) | ||||
| 			} | ||||
| 			config.WorkingDir = filepath.Join(config.WorkingDir, value) | ||||
| 		case "LABEL": | ||||
| 			// Same general idea as ENV, but we no longer allow " " | ||||
| 			// as a separator. | ||||
| 			// We didn't do that for ENV either, so nice and easy. | ||||
| 			// Potentially problematic: LABEL might theoretically | ||||
| 			// allow an = in the key? If people really do this, we | ||||
| 			// may need to investigate more advanced parsing. | ||||
| 			var ( | ||||
| 				key, val string | ||||
| 			) | ||||
|  | ||||
| 			splitLabel := strings.SplitN(value, "=", 2) | ||||
| 			// Unlike ENV, LABEL must have a value | ||||
| 			if len(splitLabel) != 2 { | ||||
| 				return nil, errors.Errorf("invalid change %q - LABEL must be formatted key=value", change) | ||||
| 			} | ||||
| 			key = splitLabel[0] | ||||
| 			val = splitLabel[1] | ||||
|  | ||||
| 			if strings.HasPrefix(key, `"`) && strings.HasSuffix(key, `"`) { | ||||
| 				key = strings.TrimPrefix(strings.TrimSuffix(key, `"`), `"`) | ||||
| 			} | ||||
| 			if strings.HasPrefix(val, `"`) && strings.HasSuffix(val, `"`) { | ||||
| 				val = strings.TrimPrefix(strings.TrimSuffix(val, `"`), `"`) | ||||
| 			} | ||||
| 			// Check key after we strip quotations | ||||
| 			if key == "" { | ||||
| 				return nil, errors.Errorf("invalid change %q - LABEL must have a non-empty key", change) | ||||
| 			} | ||||
| 			if config.Labels == nil { | ||||
| 				config.Labels = make(map[string]string) | ||||
| 			} | ||||
| 			config.Labels[key] = val | ||||
| 		case "STOPSIGNAL": | ||||
| 			// Check the provided signal for validity. | ||||
| 			killSignal, err := signal.ParseSignal(value) | ||||
| 			if err != nil { | ||||
| 				return nil, errors.Wrapf(err, "invalid change %q - KILLSIGNAL must be given a valid signal", change) | ||||
| 			} | ||||
| 			config.StopSignal = fmt.Sprintf("%d", killSignal) | ||||
| 		case "ONBUILD": | ||||
| 			// Onbuild always appends. | ||||
| 			if value == "" { | ||||
| 				return nil, errors.Errorf("invalid change %q - ONBUILD must be given an argument", change) | ||||
| 			} | ||||
| 			config.OnBuild = append(config.OnBuild, value) | ||||
| 		default: | ||||
| 			return nil, errors.Errorf("invalid change %q - invalid instruction %s", change, outerKey) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return config, nil | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 Valentin Rothberg
					Valentin Rothberg