mirror of
				https://github.com/containers/podman.git
				synced 2025-10-25 18:25:59 +08:00 
			
		
		
		
	Consolidate filter logic to pkg subdirectory
Per the conversation on pull/8724 I am consolidating filter logic and helper functions under the pkg/domain/filters dir. Signed-off-by: Baron Lenardson <lenardson.baron@gmail.com>
This commit is contained in:
		 Baron Lenardson
					Baron Lenardson
				
			
				
					committed by
					
						 Baron Lenardson
						Baron Lenardson
					
				
			
			
				
	
			
			
			 Baron Lenardson
						Baron Lenardson
					
				
			
						parent
						
							8f75ed9585
						
					
				
				
					commit
					76afb50f3a
				
			| @ -1,239 +0,0 @@ | ||||
| package lpfilters | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/containers/podman/v2/libpod" | ||||
| 	"github.com/containers/podman/v2/libpod/define" | ||||
| 	"github.com/containers/podman/v2/pkg/timetype" | ||||
| 	"github.com/containers/podman/v2/pkg/util" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
|  | ||||
| // GenerateContainerFilterFuncs return ContainerFilter functions based of filter. | ||||
| func GenerateContainerFilterFuncs(filter string, filterValues []string, r *libpod.Runtime) (func(container *libpod.Container) bool, error) { | ||||
| 	switch filter { | ||||
| 	case "id": | ||||
| 		// we only have to match one ID | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			return util.StringMatchRegexSlice(c.ID(), filterValues) | ||||
| 		}, nil | ||||
| 	case "label": | ||||
| 		// we have to match that all given labels exits on that container | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			labels := c.Labels() | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				matched := false | ||||
| 				filterArray := strings.SplitN(filterValue, "=", 2) | ||||
| 				filterKey := filterArray[0] | ||||
| 				if len(filterArray) > 1 { | ||||
| 					filterValue = filterArray[1] | ||||
| 				} else { | ||||
| 					filterValue = "" | ||||
| 				} | ||||
| 				for labelKey, labelValue := range labels { | ||||
| 					if labelKey == filterKey && ("" == filterValue || labelValue == filterValue) { | ||||
| 						matched = true | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
| 				if !matched { | ||||
| 					return false | ||||
| 				} | ||||
| 			} | ||||
| 			return true | ||||
| 		}, nil | ||||
| 	case "name": | ||||
| 		// we only have to match one name | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			return util.StringMatchRegexSlice(c.Name(), filterValues) | ||||
| 		}, nil | ||||
| 	case "exited": | ||||
| 		var exitCodes []int32 | ||||
| 		for _, exitCode := range filterValues { | ||||
| 			ec, err := strconv.ParseInt(exitCode, 10, 32) | ||||
| 			if err != nil { | ||||
| 				return nil, errors.Wrapf(err, "exited code out of range %q", ec) | ||||
| 			} | ||||
| 			exitCodes = append(exitCodes, int32(ec)) | ||||
| 		} | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			ec, exited, err := c.ExitCode() | ||||
| 			if err == nil && exited { | ||||
| 				for _, exitCode := range exitCodes { | ||||
| 					if ec == exitCode { | ||||
| 						return true | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "status": | ||||
| 		for _, filterValue := range filterValues { | ||||
| 			if !util.StringInSlice(filterValue, []string{"created", "running", "paused", "stopped", "exited", "unknown"}) { | ||||
| 				return nil, errors.Errorf("%s is not a valid status", filterValue) | ||||
| 			} | ||||
| 		} | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			status, err := c.State() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			state := status.String() | ||||
| 			if status == define.ContainerStateConfigured { | ||||
| 				state = "created" | ||||
| 			} else if status == define.ContainerStateStopped { | ||||
| 				state = "exited" | ||||
| 			} | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				if filterValue == "stopped" { | ||||
| 					filterValue = "exited" | ||||
| 				} | ||||
| 				if state == filterValue { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "ancestor": | ||||
| 		// This needs to refine to match docker | ||||
| 		// - ancestor=(<image-name>[:tag]|<image-id>| ⟨image@digest⟩) - containers created from an image or a descendant. | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				containerConfig := c.Config() | ||||
| 				if strings.Contains(containerConfig.RootfsImageID, filterValue) || strings.Contains(containerConfig.RootfsImageName, filterValue) { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "before": | ||||
| 		var createTime time.Time | ||||
| 		for _, filterValue := range filterValues { | ||||
| 			ctr, err := r.LookupContainer(filterValue) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			containerConfig := ctr.Config() | ||||
| 			if createTime.IsZero() || createTime.After(containerConfig.CreatedTime) { | ||||
| 				createTime = containerConfig.CreatedTime | ||||
| 			} | ||||
| 		} | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			cc := c.Config() | ||||
| 			return createTime.After(cc.CreatedTime) | ||||
| 		}, nil | ||||
| 	case "since": | ||||
| 		var createTime time.Time | ||||
| 		for _, filterValue := range filterValues { | ||||
| 			ctr, err := r.LookupContainer(filterValue) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			containerConfig := ctr.Config() | ||||
| 			if createTime.IsZero() || createTime.After(containerConfig.CreatedTime) { | ||||
| 				createTime = containerConfig.CreatedTime | ||||
| 			} | ||||
| 		} | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			cc := c.Config() | ||||
| 			return createTime.Before(cc.CreatedTime) | ||||
| 		}, nil | ||||
| 	case "volume": | ||||
| 		//- volume=(<volume-name>|<mount-point-destination>) | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			containerConfig := c.Config() | ||||
| 			var dest string | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				arr := strings.SplitN(filterValue, ":", 2) | ||||
| 				source := arr[0] | ||||
| 				if len(arr) == 2 { | ||||
| 					dest = arr[1] | ||||
| 				} | ||||
| 				for _, mount := range containerConfig.Spec.Mounts { | ||||
| 					if dest != "" && (mount.Source == source && mount.Destination == dest) { | ||||
| 						return true | ||||
| 					} | ||||
| 					if dest == "" && mount.Source == source { | ||||
| 						return true | ||||
| 					} | ||||
| 				} | ||||
| 				for _, vname := range containerConfig.NamedVolumes { | ||||
| 					if dest != "" && (vname.Name == source && vname.Dest == dest) { | ||||
| 						return true | ||||
| 					} | ||||
| 					if dest == "" && vname.Name == source { | ||||
| 						return true | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "health": | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			hcStatus, err := c.HealthCheckStatus() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				if hcStatus == filterValue { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "until": | ||||
| 		if len(filterValues) != 1 { | ||||
| 			return nil, errors.Errorf("specify exactly one timestamp for %s", filter) | ||||
| 		} | ||||
| 		ts, err := timetype.GetTimestamp(filterValues[0], time.Now()) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		seconds, nanoseconds, err := timetype.ParseTimestamps(ts, 0) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		until := time.Unix(seconds, nanoseconds) | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			if !until.IsZero() && c.CreatedTime().After((until)) { | ||||
| 				return true | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "pod": | ||||
| 		var pods []*libpod.Pod | ||||
| 		for _, podNameOrID := range filterValues { | ||||
| 			p, err := r.LookupPod(podNameOrID) | ||||
| 			if err != nil { | ||||
| 				if errors.Cause(err) == define.ErrNoSuchPod { | ||||
| 					continue | ||||
| 				} | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			pods = append(pods, p) | ||||
| 		} | ||||
| 		return func(c *libpod.Container) bool { | ||||
| 			// if no pods match, quick out | ||||
| 			if len(pods) < 1 { | ||||
| 				return false | ||||
| 			} | ||||
| 			// if the container has no pod id, quick out | ||||
| 			if len(c.PodID()) < 1 { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, p := range pods { | ||||
| 				// we already looked up by name or id, so id match | ||||
| 				// here is ok | ||||
| 				if p.ID() == c.PodID() { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
|  | ||||
| 	} | ||||
| 	return nil, errors.Errorf("%s is an invalid filter", filter) | ||||
| } | ||||
| @ -1,20 +0,0 @@ | ||||
| package lpfilters | ||||
|  | ||||
| import ( | ||||
| 	"net/url" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
|  | ||||
| func ParseFilterArgumentsIntoFilters(filters []string) (url.Values, error) { | ||||
| 	parsedFilters := make(url.Values) | ||||
| 	for _, f := range filters { | ||||
| 		t := strings.SplitN(f, "=", 2) | ||||
| 		if len(t) < 2 { | ||||
| 			return parsedFilters, errors.Errorf("filter input must be in the form of filter=value: %s is invalid", f) | ||||
| 		} | ||||
| 		parsedFilters.Add(t[0], t[1]) | ||||
| 	} | ||||
| 	return parsedFilters, nil | ||||
| } | ||||
| @ -1,139 +0,0 @@ | ||||
| package lpfilters | ||||
|  | ||||
| import ( | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/containers/podman/v2/libpod" | ||||
| 	"github.com/containers/podman/v2/libpod/define" | ||||
| 	"github.com/containers/podman/v2/pkg/util" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
|  | ||||
| // GeneratePodFilterFunc takes a filter and filtervalue (key, value) | ||||
| // and generates a libpod function that can be used to filter | ||||
| // pods | ||||
| func GeneratePodFilterFunc(filter string, filterValues []string) ( | ||||
| 	func(pod *libpod.Pod) bool, error) { | ||||
| 	switch filter { | ||||
| 	case "ctr-ids": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			ctrIds, err := p.AllContainersByID() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, id := range ctrIds { | ||||
| 				return util.StringMatchRegexSlice(id, filterValues) | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "ctr-names": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			ctrs, err := p.AllContainers() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, ctr := range ctrs { | ||||
| 				return util.StringMatchRegexSlice(ctr.Name(), filterValues) | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "ctr-number": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			ctrIds, err := p.AllContainersByID() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				fVint, err2 := strconv.Atoi(filterValue) | ||||
| 				if err2 != nil { | ||||
| 					return false | ||||
| 				} | ||||
| 				if len(ctrIds) == fVint { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "ctr-status": | ||||
| 		for _, filterValue := range filterValues { | ||||
| 			if !util.StringInSlice(filterValue, []string{"created", "running", "paused", "stopped", "exited", "unknown"}) { | ||||
| 				return nil, errors.Errorf("%s is not a valid status", filterValue) | ||||
| 			} | ||||
| 		} | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			ctrStatuses, err := p.Status() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, ctrStatus := range ctrStatuses { | ||||
| 				state := ctrStatus.String() | ||||
| 				if ctrStatus == define.ContainerStateConfigured { | ||||
| 					state = "created" | ||||
| 				} else if ctrStatus == define.ContainerStateStopped { | ||||
| 					state = "exited" | ||||
| 				} | ||||
| 				for _, filterValue := range filterValues { | ||||
| 					if filterValue == "stopped" { | ||||
| 						filterValue = "exited" | ||||
| 					} | ||||
| 					if state == filterValue { | ||||
| 						return true | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "id": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			return util.StringMatchRegexSlice(p.ID(), filterValues) | ||||
| 		}, nil | ||||
| 	case "name": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			return util.StringMatchRegexSlice(p.Name(), filterValues) | ||||
| 		}, nil | ||||
| 	case "status": | ||||
| 		for _, filterValue := range filterValues { | ||||
| 			if !util.StringInSlice(filterValue, []string{"stopped", "running", "paused", "exited", "dead", "created", "degraded"}) { | ||||
| 				return nil, errors.Errorf("%s is not a valid pod status", filterValue) | ||||
| 			} | ||||
| 		} | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			status, err := p.GetPodStatus() | ||||
| 			if err != nil { | ||||
| 				return false | ||||
| 			} | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				if strings.ToLower(status) == filterValue { | ||||
| 					return true | ||||
| 				} | ||||
| 			} | ||||
| 			return false | ||||
| 		}, nil | ||||
| 	case "label": | ||||
| 		return func(p *libpod.Pod) bool { | ||||
| 			labels := p.Labels() | ||||
| 			for _, filterValue := range filterValues { | ||||
| 				matched := false | ||||
| 				filterArray := strings.SplitN(filterValue, "=", 2) | ||||
| 				filterKey := filterArray[0] | ||||
| 				if len(filterArray) > 1 { | ||||
| 					filterValue = filterArray[1] | ||||
| 				} else { | ||||
| 					filterValue = "" | ||||
| 				} | ||||
| 				for labelKey, labelValue := range labels { | ||||
| 					if labelKey == filterKey && ("" == filterValue || labelValue == filterValue) { | ||||
| 						matched = true | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
| 				if !matched { | ||||
| 					return false | ||||
| 				} | ||||
| 			} | ||||
| 			return true | ||||
| 		}, nil | ||||
| 	} | ||||
| 	return nil, errors.Errorf("%s is an invalid filter", filter) | ||||
| } | ||||
		Reference in New Issue
	
	Block a user