123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- /*
- Copyright 2015 The Kubernetes Authors.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package fields
- import (
- "fmt"
- "sort"
- "strings"
- "k8s.io/kubernetes/pkg/selection"
- )
- // Selector represents a field selector.
- type Selector interface {
- // Matches returns true if this selector matches the given set of fields.
- Matches(Fields) bool
- // Empty returns true if this selector does not restrict the selection space.
- Empty() bool
- // RequiresExactMatch allows a caller to introspect whether a given selector
- // requires a single specific field to be set, and if so returns the value it
- // requires.
- RequiresExactMatch(field string) (value string, found bool)
- // Transform returns a new copy of the selector after TransformFunc has been
- // applied to the entire selector, or an error if fn returns an error.
- Transform(fn TransformFunc) (Selector, error)
- // Requirements converts this interface to Requirements to expose
- // more detailed selection information.
- Requirements() Requirements
- // String returns a human readable string that represents this selector.
- String() string
- }
- // Everything returns a selector that matches all fields.
- func Everything() Selector {
- return andTerm{}
- }
- type hasTerm struct {
- field, value string
- }
- func (t *hasTerm) Matches(ls Fields) bool {
- return ls.Get(t.field) == t.value
- }
- func (t *hasTerm) Empty() bool {
- return false
- }
- func (t *hasTerm) RequiresExactMatch(field string) (value string, found bool) {
- if t.field == field {
- return t.value, true
- }
- return "", false
- }
- func (t *hasTerm) Transform(fn TransformFunc) (Selector, error) {
- field, value, err := fn(t.field, t.value)
- if err != nil {
- return nil, err
- }
- return &hasTerm{field, value}, nil
- }
- func (t *hasTerm) Requirements() Requirements {
- return []Requirement{{
- Field: t.field,
- Operator: selection.Equals,
- Value: t.value,
- }}
- }
- func (t *hasTerm) String() string {
- return fmt.Sprintf("%v=%v", t.field, t.value)
- }
- type notHasTerm struct {
- field, value string
- }
- func (t *notHasTerm) Matches(ls Fields) bool {
- return ls.Get(t.field) != t.value
- }
- func (t *notHasTerm) Empty() bool {
- return false
- }
- func (t *notHasTerm) RequiresExactMatch(field string) (value string, found bool) {
- return "", false
- }
- func (t *notHasTerm) Transform(fn TransformFunc) (Selector, error) {
- field, value, err := fn(t.field, t.value)
- if err != nil {
- return nil, err
- }
- return ¬HasTerm{field, value}, nil
- }
- func (t *notHasTerm) Requirements() Requirements {
- return []Requirement{{
- Field: t.field,
- Operator: selection.NotEquals,
- Value: t.value,
- }}
- }
- func (t *notHasTerm) String() string {
- return fmt.Sprintf("%v!=%v", t.field, t.value)
- }
- type andTerm []Selector
- func (t andTerm) Matches(ls Fields) bool {
- for _, q := range t {
- if !q.Matches(ls) {
- return false
- }
- }
- return true
- }
- func (t andTerm) Empty() bool {
- if t == nil {
- return true
- }
- if len([]Selector(t)) == 0 {
- return true
- }
- for i := range t {
- if !t[i].Empty() {
- return false
- }
- }
- return true
- }
- func (t andTerm) RequiresExactMatch(field string) (string, bool) {
- if t == nil || len([]Selector(t)) == 0 {
- return "", false
- }
- for i := range t {
- if value, found := t[i].RequiresExactMatch(field); found {
- return value, found
- }
- }
- return "", false
- }
- func (t andTerm) Transform(fn TransformFunc) (Selector, error) {
- next := make([]Selector, len([]Selector(t)))
- for i, s := range []Selector(t) {
- n, err := s.Transform(fn)
- if err != nil {
- return nil, err
- }
- next[i] = n
- }
- return andTerm(next), nil
- }
- func (t andTerm) Requirements() Requirements {
- reqs := make([]Requirement, 0, len(t))
- for _, s := range []Selector(t) {
- rs := s.Requirements()
- reqs = append(reqs, rs...)
- }
- return reqs
- }
- func (t andTerm) String() string {
- var terms []string
- for _, q := range t {
- terms = append(terms, q.String())
- }
- return strings.Join(terms, ",")
- }
- // SelectorFromSet returns a Selector which will match exactly the given Set. A
- // nil Set is considered equivalent to Everything().
- func SelectorFromSet(ls Set) Selector {
- if ls == nil {
- return Everything()
- }
- items := make([]Selector, 0, len(ls))
- for field, value := range ls {
- items = append(items, &hasTerm{field: field, value: value})
- }
- if len(items) == 1 {
- return items[0]
- }
- return andTerm(items)
- }
- // ParseSelectorOrDie takes a string representing a selector and returns an
- // object suitable for matching, or panic when an error occur.
- func ParseSelectorOrDie(s string) Selector {
- selector, err := ParseSelector(s)
- if err != nil {
- panic(err)
- }
- return selector
- }
- // ParseSelector takes a string representing a selector and returns an
- // object suitable for matching, or an error.
- func ParseSelector(selector string) (Selector, error) {
- return parseSelector(selector,
- func(lhs, rhs string) (newLhs, newRhs string, err error) {
- return lhs, rhs, nil
- })
- }
- // Parses the selector and runs them through the given TransformFunc.
- func ParseAndTransformSelector(selector string, fn TransformFunc) (Selector, error) {
- return parseSelector(selector, fn)
- }
- // Function to transform selectors.
- type TransformFunc func(field, value string) (newField, newValue string, err error)
- func try(selectorPiece, op string) (lhs, rhs string, ok bool) {
- pieces := strings.Split(selectorPiece, op)
- if len(pieces) == 2 {
- return pieces[0], pieces[1], true
- }
- return "", "", false
- }
- func parseSelector(selector string, fn TransformFunc) (Selector, error) {
- parts := strings.Split(selector, ",")
- sort.StringSlice(parts).Sort()
- var items []Selector
- for _, part := range parts {
- if part == "" {
- continue
- }
- if lhs, rhs, ok := try(part, "!="); ok {
- items = append(items, ¬HasTerm{field: lhs, value: rhs})
- } else if lhs, rhs, ok := try(part, "=="); ok {
- items = append(items, &hasTerm{field: lhs, value: rhs})
- } else if lhs, rhs, ok := try(part, "="); ok {
- items = append(items, &hasTerm{field: lhs, value: rhs})
- } else {
- return nil, fmt.Errorf("invalid selector: '%s'; can't understand '%s'", selector, part)
- }
- }
- if len(items) == 1 {
- return items[0].Transform(fn)
- }
- return andTerm(items).Transform(fn)
- }
- // OneTermEqualSelector returns an object that matches objects where one field/field equals one value.
- // Cannot return an error.
- func OneTermEqualSelector(k, v string) Selector {
- return &hasTerm{field: k, value: v}
- }
|