diff --git a/libimage/filters.go b/libimage/filters.go index 3968c5fdc..f387edf58 100644 --- a/libimage/filters.go +++ b/libimage/filters.go @@ -159,7 +159,6 @@ func (r *Runtime) compileImageFilters(ctx context.Context, options *ListImagesOp case "label": filter = filterLabel(ctx, value) - case "readonly": readOnly, err := r.bool(duplicate, key, value) if err != nil { diff --git a/libnetwork/util/filters.go b/libnetwork/util/filters.go index a8ef8a413..2f1e4a21f 100644 --- a/libnetwork/util/filters.go +++ b/libnetwork/util/filters.go @@ -65,7 +65,10 @@ func createPruneFilterFuncs(key string, filterValues []string) (types.FilterFunc return func(net types.Network) bool { return filters.MatchLabelFilters(filterValues, net.Labels) }, nil - + case "label!": + return func(net types.Network) bool { + return !filters.MatchLabelFilters(filterValues, net.Labels) + }, nil case "until": until, err := filters.ComputeUntilTimestamp(filterValues) if err != nil { diff --git a/libnetwork/util/filters_test.go b/libnetwork/util/filters_test.go new file mode 100644 index 000000000..cb1e09986 --- /dev/null +++ b/libnetwork/util/filters_test.go @@ -0,0 +1,47 @@ +package util + +import "testing" + +func TestGenerateFilterFunc(t *testing.T) { + testValues := []string{ + "", + "test", + "", + } + type args struct { + keys []string + labels []string + } + tests := []struct { + name string + args args + want bool + }{ + { + name: "Match when all filters the same as labels", + args: args{ + keys: []string{"label", "label", "label"}, + labels: testValues, + }, + want: true, + }, + { + name: "Match with inverse", + args: args{ + keys: []string{"label", "label", "label!"}, + labels: testValues, + }, + want: false, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + for _, entry := range tt.args.keys { + if _, err := createFilterFuncs(entry, tt.args.labels); err != nil { + t.Errorf("createPruneFilterFuncs() failed on %s with entry %s: %s", tt.name, entry, err.Error()) + } + } + }) + } +} diff --git a/pkg/filters/filters.go b/pkg/filters/filters.go index dee66107f..f8b0066e5 100644 --- a/pkg/filters/filters.go +++ b/pkg/filters/filters.go @@ -4,6 +4,7 @@ import ( "encoding/json" "fmt" "net/http" + "path/filepath" "strings" "time" @@ -111,11 +112,24 @@ outer: filterValue = "" } for labelKey, labelValue := range labels { - if labelKey == filterKey && (filterValue == "" || labelValue == filterValue) { - continue outer + if filterValue == "" || labelValue == filterValue { + if labelKey == filterKey || matchPattern(filterKey, labelKey) { + continue outer + } } } return false } return true } + +func matchPattern(pattern string, value string) bool { + if strings.Contains(pattern, "*") { + filter := fmt.Sprintf("*%s*", pattern) + filter = strings.ReplaceAll(filter, string(filepath.Separator), "|") + newName := strings.ReplaceAll(value, string(filepath.Separator), "|") + match, _ := filepath.Match(filter, newName) + return match + } + return false +}