mirror of
https://github.com/kyverno/kyverno.git
synced 2025-03-06 16:06:56 +00:00
89 lines
2.2 KiB
Go
89 lines
2.2 KiB
Go
package context
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
|
|
datautils "github.com/kyverno/kyverno/pkg/utils/data"
|
|
)
|
|
|
|
// Query the JSON context with JMESPATH search path
|
|
func (ctx *context) Query(query string) (interface{}, error) {
|
|
if err := ctx.loadDeferred(query); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
query = strings.TrimSpace(query)
|
|
if query == "" {
|
|
return nil, fmt.Errorf("invalid query (nil)")
|
|
}
|
|
// compile the query
|
|
queryPath, err := ctx.jp.Query(query)
|
|
if err != nil {
|
|
logger.Error(err, "incorrect query", "query", query)
|
|
return nil, fmt.Errorf("incorrect query %s: %v", query, err)
|
|
}
|
|
// search
|
|
ctx.mutex.RLock()
|
|
defer ctx.mutex.RUnlock()
|
|
var data interface{}
|
|
if err := json.Unmarshal(ctx.jsonRaw, &data); err != nil {
|
|
return nil, fmt.Errorf("failed to unmarshal context: %w", err)
|
|
}
|
|
result, err := queryPath.Search(data)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("JMESPath query failed: %w", err)
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
func (ctx *context) loadDeferred(query string) error {
|
|
loaders := ctx.getMatchingLoaders(query)
|
|
for _, loader := range loaders {
|
|
err := ctx.evaluateLoader(loader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (ctx *context) getMatchingLoaders(query string) []string {
|
|
ctx.deferred.mutex.Lock()
|
|
defer ctx.deferred.mutex.Unlock()
|
|
var matchingLoaders []string
|
|
for name := range ctx.deferred.loaders {
|
|
if strings.Contains(query, name) {
|
|
matchingLoaders = append(matchingLoaders, name)
|
|
}
|
|
}
|
|
return matchingLoaders
|
|
}
|
|
|
|
func (ctx *context) evaluateLoader(name string) error {
|
|
loader, ok := ctx.deferred.loaders[name]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
delete(ctx.deferred.loaders, name)
|
|
return loader()
|
|
}
|
|
|
|
func (ctx *context) HasChanged(jmespath string) (bool, error) {
|
|
objData, err := ctx.Query("request.object." + jmespath)
|
|
if err != nil {
|
|
return false, fmt.Errorf("failed to query request.object: %w", err)
|
|
}
|
|
if objData == nil {
|
|
return false, fmt.Errorf("request.object.%s not found", jmespath)
|
|
}
|
|
oldObjData, err := ctx.Query("request.oldObject." + jmespath)
|
|
if err != nil {
|
|
return false, fmt.Errorf("failed to query request.object: %w", err)
|
|
}
|
|
if oldObjData == nil {
|
|
return false, fmt.Errorf("request.oldObject.%s not found", jmespath)
|
|
}
|
|
return !datautils.DeepEqual(objData, oldObjData), nil
|
|
}
|