Files
kubesphere/vendor/github.com/open-policy-agent/opa/topdown/walk.go
2023-02-12 23:09:20 +08:00

97 lines
2.1 KiB
Go

// Copyright 2017 The OPA Authors. All rights reserved.
// Use of this source code is governed by an Apache2
// license that can be found in the LICENSE file.
package topdown
import (
"github.com/open-policy-agent/opa/ast"
)
func evalWalk(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error {
input := operands[0]
filter := getOutputPath(operands)
return walk(filter, nil, input, iter)
}
func walk(filter, path *ast.Array, input *ast.Term, iter func(*ast.Term) error) error {
if filter == nil || filter.Len() == 0 {
if path == nil {
path = ast.NewArray()
}
if err := iter(ast.ArrayTerm(ast.NewTerm(path.Copy()), input)); err != nil {
return err
}
}
if filter != nil && filter.Len() > 0 {
key := filter.Elem(0)
filter = filter.Slice(1, -1)
if key.IsGround() {
if term := input.Get(key); term != nil {
path = pathAppend(path, key)
return walk(filter, path, term, iter)
}
return nil
}
}
switch v := input.Value.(type) {
case *ast.Array:
for i := 0; i < v.Len(); i++ {
path = pathAppend(path, ast.IntNumberTerm(i))
if err := walk(filter, path, v.Elem(i), iter); err != nil {
return err
}
path = path.Slice(0, path.Len()-1)
}
case ast.Object:
return v.Iter(func(k, v *ast.Term) error {
path = pathAppend(path, k)
if err := walk(filter, path, v, iter); err != nil {
return err
}
path = path.Slice(0, path.Len()-1)
return nil
})
case ast.Set:
return v.Iter(func(elem *ast.Term) error {
path = pathAppend(path, elem)
if err := walk(filter, path, elem, iter); err != nil {
return err
}
path = path.Slice(0, path.Len()-1)
return nil
})
}
return nil
}
func pathAppend(path *ast.Array, key *ast.Term) *ast.Array {
if path == nil {
return ast.NewArray(key)
}
return path.Append(key)
}
func getOutputPath(operands []*ast.Term) *ast.Array {
if len(operands) == 2 {
if arr, ok := operands[1].Value.(*ast.Array); ok {
if arr.Len() == 2 {
if path, ok := arr.Elem(0).Value.(*ast.Array); ok {
return path
}
}
}
}
return nil
}
func init() {
RegisterBuiltinFunc(ast.WalkBuiltin.Name, evalWalk)
}