mirror of
https://github.com/status-im/consul.git
synced 2025-01-12 06:44:41 +00:00
afa1cc98d1
Fixes: #4222 # Data Filtering This PR will implement filtering for the following endpoints: ## Supported HTTP Endpoints - `/agent/checks` - `/agent/services` - `/catalog/nodes` - `/catalog/service/:service` - `/catalog/connect/:service` - `/catalog/node/:node` - `/health/node/:node` - `/health/checks/:service` - `/health/service/:service` - `/health/connect/:service` - `/health/state/:state` - `/internal/ui/nodes` - `/internal/ui/services` More can be added going forward and any endpoint which is used to list some data is a good candidate. ## Usage When using the HTTP API a `filter` query parameter can be used to pass a filter expression to Consul. Filter Expressions take the general form of: ``` <selector> == <value> <selector> != <value> <value> in <selector> <value> not in <selector> <selector> contains <value> <selector> not contains <value> <selector> is empty <selector> is not empty not <other expression> <expression 1> and <expression 2> <expression 1> or <expression 2> ``` Normal boolean logic and precedence is supported. All of the actual filtering and evaluation logic is coming from the [go-bexpr](https://github.com/hashicorp/go-bexpr) library ## Other changes Adding the `Internal.ServiceDump` RPC endpoint. This will allow the UI to filter services better.
149 lines
3.7 KiB
Go
149 lines
3.7 KiB
Go
package objx
|
|
|
|
import (
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
// arrayAccesRegexString is the regex used to extract the array number
|
|
// from the access path
|
|
const arrayAccesRegexString = `^(.+)\[([0-9]+)\]$`
|
|
|
|
// arrayAccesRegex is the compiled arrayAccesRegexString
|
|
var arrayAccesRegex = regexp.MustCompile(arrayAccesRegexString)
|
|
|
|
// Get gets the value using the specified selector and
|
|
// returns it inside a new Obj object.
|
|
//
|
|
// If it cannot find the value, Get will return a nil
|
|
// value inside an instance of Obj.
|
|
//
|
|
// Get can only operate directly on map[string]interface{} and []interface.
|
|
//
|
|
// Example
|
|
//
|
|
// To access the title of the third chapter of the second book, do:
|
|
//
|
|
// o.Get("books[1].chapters[2].title")
|
|
func (m Map) Get(selector string) *Value {
|
|
rawObj := access(m, selector, nil, false)
|
|
return &Value{data: rawObj}
|
|
}
|
|
|
|
// Set sets the value using the specified selector and
|
|
// returns the object on which Set was called.
|
|
//
|
|
// Set can only operate directly on map[string]interface{} and []interface
|
|
//
|
|
// Example
|
|
//
|
|
// To set the title of the third chapter of the second book, do:
|
|
//
|
|
// o.Set("books[1].chapters[2].title","Time to Go")
|
|
func (m Map) Set(selector string, value interface{}) Map {
|
|
access(m, selector, value, true)
|
|
return m
|
|
}
|
|
|
|
// access accesses the object using the selector and performs the
|
|
// appropriate action.
|
|
func access(current, selector, value interface{}, isSet bool) interface{} {
|
|
switch selector.(type) {
|
|
case int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64:
|
|
if array, ok := current.([]interface{}); ok {
|
|
index := intFromInterface(selector)
|
|
if index >= len(array) {
|
|
return nil
|
|
}
|
|
return array[index]
|
|
}
|
|
return nil
|
|
|
|
case string:
|
|
selStr := selector.(string)
|
|
selSegs := strings.SplitN(selStr, PathSeparator, 2)
|
|
thisSel := selSegs[0]
|
|
index := -1
|
|
var err error
|
|
|
|
if strings.Contains(thisSel, "[") {
|
|
arrayMatches := arrayAccesRegex.FindStringSubmatch(thisSel)
|
|
if len(arrayMatches) > 0 {
|
|
// Get the key into the map
|
|
thisSel = arrayMatches[1]
|
|
|
|
// Get the index into the array at the key
|
|
index, err = strconv.Atoi(arrayMatches[2])
|
|
|
|
if err != nil {
|
|
// This should never happen. If it does, something has gone
|
|
// seriously wrong. Panic.
|
|
panic("objx: Array index is not an integer. Must use array[int].")
|
|
}
|
|
}
|
|
}
|
|
if curMap, ok := current.(Map); ok {
|
|
current = map[string]interface{}(curMap)
|
|
}
|
|
// get the object in question
|
|
switch current.(type) {
|
|
case map[string]interface{}:
|
|
curMSI := current.(map[string]interface{})
|
|
if len(selSegs) <= 1 && isSet {
|
|
curMSI[thisSel] = value
|
|
return nil
|
|
}
|
|
current = curMSI[thisSel]
|
|
default:
|
|
current = nil
|
|
}
|
|
// do we need to access the item of an array?
|
|
if index > -1 {
|
|
if array, ok := current.([]interface{}); ok {
|
|
if index < len(array) {
|
|
current = array[index]
|
|
} else {
|
|
current = nil
|
|
}
|
|
}
|
|
}
|
|
if len(selSegs) > 1 {
|
|
current = access(current, selSegs[1], value, isSet)
|
|
}
|
|
}
|
|
return current
|
|
}
|
|
|
|
// intFromInterface converts an interface object to the largest
|
|
// representation of an unsigned integer using a type switch and
|
|
// assertions
|
|
func intFromInterface(selector interface{}) int {
|
|
var value int
|
|
switch selector.(type) {
|
|
case int:
|
|
value = selector.(int)
|
|
case int8:
|
|
value = int(selector.(int8))
|
|
case int16:
|
|
value = int(selector.(int16))
|
|
case int32:
|
|
value = int(selector.(int32))
|
|
case int64:
|
|
value = int(selector.(int64))
|
|
case uint:
|
|
value = int(selector.(uint))
|
|
case uint8:
|
|
value = int(selector.(uint8))
|
|
case uint16:
|
|
value = int(selector.(uint16))
|
|
case uint32:
|
|
value = int(selector.(uint32))
|
|
case uint64:
|
|
value = int(selector.(uint64))
|
|
default:
|
|
return 0
|
|
}
|
|
return value
|
|
}
|