1998 lines
52 KiB
Go
1998 lines
52 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 rego exposes high level APIs for evaluating Rego policies.
|
|
package rego
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
|
|
"github.com/open-policy-agent/opa/loader"
|
|
"github.com/open-policy-agent/opa/types"
|
|
|
|
"github.com/open-policy-agent/opa/bundle"
|
|
|
|
"github.com/open-policy-agent/opa/ast"
|
|
"github.com/open-policy-agent/opa/internal/compiler/wasm"
|
|
"github.com/open-policy-agent/opa/internal/ir"
|
|
"github.com/open-policy-agent/opa/internal/planner"
|
|
"github.com/open-policy-agent/opa/internal/wasm/encoding"
|
|
"github.com/open-policy-agent/opa/metrics"
|
|
"github.com/open-policy-agent/opa/storage"
|
|
"github.com/open-policy-agent/opa/storage/inmem"
|
|
"github.com/open-policy-agent/opa/topdown"
|
|
"github.com/open-policy-agent/opa/util"
|
|
)
|
|
|
|
const defaultPartialNamespace = "partial"
|
|
|
|
// CompileResult represents the result of compiling a Rego query, zero or more
|
|
// Rego modules, and arbitrary contextual data into an executable.
|
|
type CompileResult struct {
|
|
Bytes []byte `json:"bytes"`
|
|
}
|
|
|
|
// PartialQueries contains the queries and support modules produced by partial
|
|
// evaluation.
|
|
type PartialQueries struct {
|
|
Queries []ast.Body `json:"queries,omitempty"`
|
|
Support []*ast.Module `json:"modules,omitempty"`
|
|
}
|
|
|
|
// PartialResult represents the result of partial evaluation. The result can be
|
|
// used to generate a new query that can be run when inputs are known.
|
|
type PartialResult struct {
|
|
compiler *ast.Compiler
|
|
store storage.Store
|
|
body ast.Body
|
|
builtinDecls map[string]*ast.Builtin
|
|
builtinFuncs map[string]*topdown.Builtin
|
|
}
|
|
|
|
// Rego returns an object that can be evaluated to produce a query result.
|
|
func (pr PartialResult) Rego(options ...func(*Rego)) *Rego {
|
|
options = append(options, Compiler(pr.compiler), Store(pr.store), ParsedQuery(pr.body))
|
|
r := New(options...)
|
|
|
|
// Propagate any custom builtins.
|
|
for k, v := range pr.builtinDecls {
|
|
r.builtinDecls[k] = v
|
|
}
|
|
for k, v := range pr.builtinFuncs {
|
|
r.builtinFuncs[k] = v
|
|
}
|
|
return r
|
|
}
|
|
|
|
// preparedQuery is a wrapper around a Rego object which has pre-processed
|
|
// state stored on it. Once prepared there are a more limited number of actions
|
|
// that can be taken with it. It will, however, be able to evaluate faster since
|
|
// it will not have to re-parse or compile as much.
|
|
type preparedQuery struct {
|
|
r *Rego
|
|
cfg *PrepareConfig
|
|
}
|
|
|
|
// EvalContext defines the set of options allowed to be set at evaluation
|
|
// time. Any other options will need to be set on a new Rego object.
|
|
type EvalContext struct {
|
|
hasInput bool
|
|
rawInput *interface{}
|
|
parsedInput ast.Value
|
|
metrics metrics.Metrics
|
|
txn storage.Transaction
|
|
instrument bool
|
|
instrumentation *topdown.Instrumentation
|
|
partialNamespace string
|
|
tracers []topdown.Tracer
|
|
compiledQuery compiledQuery
|
|
unknowns []string
|
|
disableInlining []ast.Ref
|
|
parsedUnknowns []*ast.Term
|
|
indexing bool
|
|
}
|
|
|
|
// EvalOption defines a function to set an option on an EvalConfig
|
|
type EvalOption func(*EvalContext)
|
|
|
|
// EvalInput configures the input for a Prepared Query's evaluation
|
|
func EvalInput(input interface{}) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.rawInput = &input
|
|
e.hasInput = true
|
|
}
|
|
}
|
|
|
|
// EvalParsedInput configures the input for a Prepared Query's evaluation
|
|
func EvalParsedInput(input ast.Value) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.parsedInput = input
|
|
e.hasInput = true
|
|
}
|
|
}
|
|
|
|
// EvalMetrics configures the metrics for a Prepared Query's evaluation
|
|
func EvalMetrics(metric metrics.Metrics) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.metrics = metric
|
|
}
|
|
}
|
|
|
|
// EvalTransaction configures the Transaction for a Prepared Query's evaluation
|
|
func EvalTransaction(txn storage.Transaction) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.txn = txn
|
|
}
|
|
}
|
|
|
|
// EvalInstrument enables or disables instrumenting for a Prepared Query's evaluation
|
|
func EvalInstrument(instrument bool) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.instrument = instrument
|
|
}
|
|
}
|
|
|
|
// EvalTracer configures a tracer for a Prepared Query's evaluation
|
|
func EvalTracer(tracer topdown.Tracer) EvalOption {
|
|
return func(e *EvalContext) {
|
|
if tracer != nil {
|
|
e.tracers = append(e.tracers, tracer)
|
|
}
|
|
}
|
|
}
|
|
|
|
// EvalPartialNamespace returns an argument that sets the namespace to use for
|
|
// partial evaluation results. The namespace must be a valid package path
|
|
// component.
|
|
func EvalPartialNamespace(ns string) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.partialNamespace = ns
|
|
}
|
|
}
|
|
|
|
// EvalUnknowns returns an argument that sets the values to treat as
|
|
// unknown during partial evaluation.
|
|
func EvalUnknowns(unknowns []string) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.unknowns = unknowns
|
|
}
|
|
}
|
|
|
|
// EvalDisableInlining returns an argument that adds a set of paths to exclude from
|
|
// partial evaluation inlining.
|
|
func EvalDisableInlining(paths []ast.Ref) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.disableInlining = paths
|
|
}
|
|
}
|
|
|
|
// EvalParsedUnknowns returns an argument that sets the values to treat
|
|
// as unknown during partial evaluation.
|
|
func EvalParsedUnknowns(unknowns []*ast.Term) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.parsedUnknowns = unknowns
|
|
}
|
|
}
|
|
|
|
// EvalRuleIndexing will disable indexing optimizations for the
|
|
// evaluation. This should only be used when tracing in debug mode.
|
|
func EvalRuleIndexing(enabled bool) EvalOption {
|
|
return func(e *EvalContext) {
|
|
e.indexing = enabled
|
|
}
|
|
}
|
|
|
|
func (pq preparedQuery) Modules() map[string]*ast.Module {
|
|
mods := make(map[string]*ast.Module)
|
|
|
|
for name, mod := range pq.r.parsedModules {
|
|
mods[name] = mod
|
|
}
|
|
|
|
for path, b := range pq.r.bundles {
|
|
for name, mod := range b.ParsedModules(path) {
|
|
mods[name] = mod
|
|
}
|
|
}
|
|
|
|
return mods
|
|
}
|
|
|
|
// newEvalContext creates a new EvalContext overlaying any EvalOptions over top
|
|
// the Rego object on the preparedQuery. The returned function should be called
|
|
// once the evaluation is complete to close any transactions that might have
|
|
// been opened.
|
|
func (pq preparedQuery) newEvalContext(ctx context.Context, options []EvalOption) (*EvalContext, func(context.Context), error) {
|
|
ectx := &EvalContext{
|
|
hasInput: false,
|
|
rawInput: nil,
|
|
parsedInput: nil,
|
|
metrics: nil,
|
|
txn: nil,
|
|
instrument: false,
|
|
instrumentation: nil,
|
|
partialNamespace: pq.r.partialNamespace,
|
|
tracers: nil,
|
|
unknowns: pq.r.unknowns,
|
|
parsedUnknowns: pq.r.parsedUnknowns,
|
|
compiledQuery: compiledQuery{},
|
|
indexing: true,
|
|
}
|
|
|
|
for _, o := range options {
|
|
o(ectx)
|
|
}
|
|
|
|
if ectx.metrics == nil {
|
|
ectx.metrics = metrics.New()
|
|
}
|
|
|
|
if ectx.instrument {
|
|
ectx.instrumentation = topdown.NewInstrumentation(ectx.metrics)
|
|
}
|
|
|
|
// Default to an empty "finish" function
|
|
finishFunc := func(context.Context) {}
|
|
|
|
var err error
|
|
ectx.disableInlining, err = parseStringsToRefs(pq.r.disableInlining)
|
|
if err != nil {
|
|
return nil, finishFunc, err
|
|
}
|
|
|
|
if ectx.txn == nil {
|
|
ectx.txn, err = pq.r.store.NewTransaction(ctx)
|
|
if err != nil {
|
|
return nil, finishFunc, err
|
|
}
|
|
finishFunc = func(ctx context.Context) {
|
|
pq.r.store.Abort(ctx, ectx.txn)
|
|
}
|
|
}
|
|
|
|
// If we didn't get an input specified in the Eval options
|
|
// then fall back to the Rego object's input fields.
|
|
if !ectx.hasInput {
|
|
ectx.rawInput = pq.r.rawInput
|
|
ectx.parsedInput = pq.r.parsedInput
|
|
}
|
|
|
|
if ectx.parsedInput == nil {
|
|
if ectx.rawInput == nil {
|
|
// Fall back to the original Rego objects input if none was specified
|
|
// Note that it could still be nil
|
|
ectx.rawInput = pq.r.rawInput
|
|
}
|
|
ectx.parsedInput, err = pq.r.parseRawInput(ectx.rawInput, ectx.metrics)
|
|
if err != nil {
|
|
return nil, finishFunc, err
|
|
}
|
|
}
|
|
|
|
return ectx, finishFunc, nil
|
|
}
|
|
|
|
// PreparedEvalQuery holds the prepared Rego state that has been pre-processed
|
|
// for subsequent evaluations.
|
|
type PreparedEvalQuery struct {
|
|
preparedQuery
|
|
}
|
|
|
|
// Eval evaluates this PartialResult's Rego object with additional eval options
|
|
// and returns a ResultSet.
|
|
// If options are provided they will override the original Rego options respective value.
|
|
// The original Rego object transaction will *not* be re-used. A new transaction will be opened
|
|
// if one is not provided with an EvalOption.
|
|
func (pq PreparedEvalQuery) Eval(ctx context.Context, options ...EvalOption) (ResultSet, error) {
|
|
ectx, finish, err := pq.newEvalContext(ctx, options)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer finish(ctx)
|
|
|
|
ectx.compiledQuery = pq.r.compiledQueries[evalQueryType]
|
|
|
|
return pq.r.eval(ctx, ectx)
|
|
}
|
|
|
|
// PreparedPartialQuery holds the prepared Rego state that has been pre-processed
|
|
// for partial evaluations.
|
|
type PreparedPartialQuery struct {
|
|
preparedQuery
|
|
}
|
|
|
|
// Partial runs partial evaluation on the prepared query and returns the result.
|
|
// The original Rego object transaction will *not* be re-used. A new transaction will be opened
|
|
// if one is not provided with an EvalOption.
|
|
func (pq PreparedPartialQuery) Partial(ctx context.Context, options ...EvalOption) (*PartialQueries, error) {
|
|
ectx, finish, err := pq.newEvalContext(ctx, options)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer finish(ctx)
|
|
|
|
ectx.compiledQuery = pq.r.compiledQueries[partialQueryType]
|
|
|
|
return pq.r.partial(ctx, ectx)
|
|
}
|
|
|
|
// Result defines the output of Rego evaluation.
|
|
type Result struct {
|
|
Expressions []*ExpressionValue `json:"expressions"`
|
|
Bindings Vars `json:"bindings,omitempty"`
|
|
}
|
|
|
|
func newResult() Result {
|
|
return Result{
|
|
Bindings: Vars{},
|
|
}
|
|
}
|
|
|
|
// Location defines a position in a Rego query or module.
|
|
type Location struct {
|
|
Row int `json:"row"`
|
|
Col int `json:"col"`
|
|
}
|
|
|
|
// ExpressionValue defines the value of an expression in a Rego query.
|
|
type ExpressionValue struct {
|
|
Value interface{} `json:"value"`
|
|
Text string `json:"text"`
|
|
Location *Location `json:"location"`
|
|
}
|
|
|
|
func newExpressionValue(expr *ast.Expr, value interface{}) *ExpressionValue {
|
|
result := &ExpressionValue{
|
|
Value: value,
|
|
}
|
|
if expr.Location != nil {
|
|
result.Text = string(expr.Location.Text)
|
|
result.Location = &Location{
|
|
Row: expr.Location.Row,
|
|
Col: expr.Location.Col,
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (ev *ExpressionValue) String() string {
|
|
return fmt.Sprint(ev.Value)
|
|
}
|
|
|
|
// ResultSet represents a collection of output from Rego evaluation. An empty
|
|
// result set represents an undefined query.
|
|
type ResultSet []Result
|
|
|
|
// Vars represents a collection of variable bindings. The keys are the variable
|
|
// names and the values are the binding values.
|
|
type Vars map[string]interface{}
|
|
|
|
// WithoutWildcards returns a copy of v with wildcard variables removed.
|
|
func (v Vars) WithoutWildcards() Vars {
|
|
n := Vars{}
|
|
for k, v := range v {
|
|
if ast.Var(k).IsWildcard() || ast.Var(k).IsGenerated() {
|
|
continue
|
|
}
|
|
n[k] = v
|
|
}
|
|
return n
|
|
}
|
|
|
|
// Errors represents a collection of errors returned when evaluating Rego.
|
|
type Errors []error
|
|
|
|
func (errs Errors) Error() string {
|
|
if len(errs) == 0 {
|
|
return "no error"
|
|
}
|
|
if len(errs) == 1 {
|
|
return fmt.Sprintf("1 error occurred: %v", errs[0].Error())
|
|
}
|
|
buf := []string{fmt.Sprintf("%v errors occurred", len(errs))}
|
|
for _, err := range errs {
|
|
buf = append(buf, err.Error())
|
|
}
|
|
return strings.Join(buf, "\n")
|
|
}
|
|
|
|
type compiledQuery struct {
|
|
query ast.Body
|
|
compiler ast.QueryCompiler
|
|
}
|
|
|
|
type queryType int
|
|
|
|
// Define a query type for each of the top level Rego
|
|
// API's that compile queries differently.
|
|
const (
|
|
evalQueryType queryType = iota
|
|
partialResultQueryType queryType = iota
|
|
partialQueryType queryType = iota
|
|
compileQueryType queryType = iota
|
|
)
|
|
|
|
type loadPaths struct {
|
|
paths []string
|
|
filter loader.Filter
|
|
}
|
|
|
|
// Rego constructs a query and can be evaluated to obtain results.
|
|
type Rego struct {
|
|
query string
|
|
parsedQuery ast.Body
|
|
compiledQueries map[queryType]compiledQuery
|
|
pkg string
|
|
parsedPackage *ast.Package
|
|
imports []string
|
|
parsedImports []*ast.Import
|
|
rawInput *interface{}
|
|
parsedInput ast.Value
|
|
unknowns []string
|
|
parsedUnknowns []*ast.Term
|
|
disableInlining []string
|
|
partialNamespace string
|
|
modules []rawModule
|
|
parsedModules map[string]*ast.Module
|
|
compiler *ast.Compiler
|
|
store storage.Store
|
|
ownStore bool
|
|
txn storage.Transaction
|
|
metrics metrics.Metrics
|
|
tracers []topdown.Tracer
|
|
tracebuf *topdown.BufferTracer
|
|
trace bool
|
|
instrumentation *topdown.Instrumentation
|
|
instrument bool
|
|
capture map[*ast.Expr]ast.Var // map exprs to generated capture vars
|
|
termVarID int
|
|
dump io.Writer
|
|
runtime *ast.Term
|
|
builtinDecls map[string]*ast.Builtin
|
|
builtinFuncs map[string]*topdown.Builtin
|
|
unsafeBuiltins map[string]struct{}
|
|
loadPaths loadPaths
|
|
bundlePaths []string
|
|
bundles map[string]*bundle.Bundle
|
|
}
|
|
|
|
// Function represents a built-in function that is callable in Rego.
|
|
type Function struct {
|
|
Name string
|
|
Decl *types.Function
|
|
Memoize bool
|
|
}
|
|
|
|
// BuiltinContext contains additional attributes from the evaluator that
|
|
// built-in functions can use, e.g., the request context.Context, caches, etc.
|
|
type BuiltinContext = topdown.BuiltinContext
|
|
|
|
type (
|
|
// Builtin1 defines a built-in function that accepts 1 argument.
|
|
Builtin1 func(bctx BuiltinContext, op1 *ast.Term) (*ast.Term, error)
|
|
|
|
// Builtin2 defines a built-in function that accepts 2 arguments.
|
|
Builtin2 func(bctx BuiltinContext, op1, op2 *ast.Term) (*ast.Term, error)
|
|
|
|
// Builtin3 defines a built-in function that accepts 3 argument.
|
|
Builtin3 func(bctx BuiltinContext, op1, op2, op3 *ast.Term) (*ast.Term, error)
|
|
|
|
// Builtin4 defines a built-in function that accepts 4 argument.
|
|
Builtin4 func(bctx BuiltinContext, op1, op2, op3, op4 *ast.Term) (*ast.Term, error)
|
|
|
|
// BuiltinDyn defines a built-in function that accepts a list of arguments.
|
|
BuiltinDyn func(bctx BuiltinContext, terms []*ast.Term) (*ast.Term, error)
|
|
)
|
|
|
|
// Function1 returns an option that adds a built-in function to the Rego object.
|
|
func Function1(decl *Function, f Builtin1) func(*Rego) {
|
|
return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error {
|
|
result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0]) })
|
|
return finishFunction(decl.Name, bctx, result, err, iter)
|
|
})
|
|
}
|
|
|
|
// Function2 returns an option that adds a built-in function to the Rego object.
|
|
func Function2(decl *Function, f Builtin2) func(*Rego) {
|
|
return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error {
|
|
result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1]) })
|
|
return finishFunction(decl.Name, bctx, result, err, iter)
|
|
})
|
|
}
|
|
|
|
// Function3 returns an option that adds a built-in function to the Rego object.
|
|
func Function3(decl *Function, f Builtin3) func(*Rego) {
|
|
return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error {
|
|
result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1], terms[2]) })
|
|
return finishFunction(decl.Name, bctx, result, err, iter)
|
|
})
|
|
}
|
|
|
|
// Function4 returns an option that adds a built-in function to the Rego object.
|
|
func Function4(decl *Function, f Builtin4) func(*Rego) {
|
|
return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error {
|
|
result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1], terms[2], terms[3]) })
|
|
return finishFunction(decl.Name, bctx, result, err, iter)
|
|
})
|
|
}
|
|
|
|
// FunctionDyn returns an option that adds a built-in function to the Rego object.
|
|
func FunctionDyn(decl *Function, f BuiltinDyn) func(*Rego) {
|
|
return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error {
|
|
result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms) })
|
|
return finishFunction(decl.Name, bctx, result, err, iter)
|
|
})
|
|
}
|
|
|
|
// FunctionDecl returns an option that adds a custom-built-in function
|
|
// __declaration__. NO implementation is provided. This is used for
|
|
// non-interpreter execution envs (e.g., Wasm).
|
|
func FunctionDecl(decl *Function) func(*Rego) {
|
|
return newDecl(decl)
|
|
}
|
|
|
|
func newDecl(decl *Function) func(*Rego) {
|
|
return func(r *Rego) {
|
|
r.builtinDecls[decl.Name] = &ast.Builtin{
|
|
Name: decl.Name,
|
|
Decl: decl.Decl,
|
|
}
|
|
}
|
|
}
|
|
|
|
type memo struct {
|
|
term *ast.Term
|
|
err error
|
|
}
|
|
|
|
type memokey string
|
|
|
|
func memoize(decl *Function, bctx BuiltinContext, terms []*ast.Term, ifEmpty func() (*ast.Term, error)) (*ast.Term, error) {
|
|
|
|
if !decl.Memoize {
|
|
return ifEmpty()
|
|
}
|
|
|
|
// NOTE(tsandall): we assume memoization is applied to infrequent built-in
|
|
// calls that do things like fetch data from remote locations. As such,
|
|
// converting the terms to strings is acceptable for now.
|
|
var b strings.Builder
|
|
if _, err := b.WriteString(decl.Name); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// The term slice _may_ include an output term depending on how the caller
|
|
// referred to the built-in function. Only use the arguments as the cache
|
|
// key. Unification ensures we don't get false positive matches.
|
|
for i := 0; i < len(decl.Decl.Args()); i++ {
|
|
if _, err := b.WriteString(terms[i].String()); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
key := memokey(b.String())
|
|
hit, ok := bctx.Cache.Get(key)
|
|
var m memo
|
|
if ok {
|
|
m = hit.(memo)
|
|
} else {
|
|
m.term, m.err = ifEmpty()
|
|
bctx.Cache.Put(key, m)
|
|
}
|
|
|
|
return m.term, m.err
|
|
}
|
|
|
|
// Dump returns an argument that sets the writer to dump debugging information to.
|
|
func Dump(w io.Writer) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.dump = w
|
|
}
|
|
}
|
|
|
|
// Query returns an argument that sets the Rego query.
|
|
func Query(q string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.query = q
|
|
}
|
|
}
|
|
|
|
// ParsedQuery returns an argument that sets the Rego query.
|
|
func ParsedQuery(q ast.Body) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.parsedQuery = q
|
|
}
|
|
}
|
|
|
|
// Package returns an argument that sets the Rego package on the query's
|
|
// context.
|
|
func Package(p string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.pkg = p
|
|
}
|
|
}
|
|
|
|
// ParsedPackage returns an argument that sets the Rego package on the query's
|
|
// context.
|
|
func ParsedPackage(pkg *ast.Package) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.parsedPackage = pkg
|
|
}
|
|
}
|
|
|
|
// Imports returns an argument that adds a Rego import to the query's context.
|
|
func Imports(p []string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.imports = append(r.imports, p...)
|
|
}
|
|
}
|
|
|
|
// ParsedImports returns an argument that adds Rego imports to the query's
|
|
// context.
|
|
func ParsedImports(imp []*ast.Import) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.parsedImports = append(r.parsedImports, imp...)
|
|
}
|
|
}
|
|
|
|
// Input returns an argument that sets the Rego input document. Input should be
|
|
// a native Go value representing the input document.
|
|
func Input(x interface{}) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.rawInput = &x
|
|
}
|
|
}
|
|
|
|
// ParsedInput returns an argument that sets the Rego input document.
|
|
func ParsedInput(x ast.Value) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.parsedInput = x
|
|
}
|
|
}
|
|
|
|
// Unknowns returns an argument that sets the values to treat as unknown during
|
|
// partial evaluation.
|
|
func Unknowns(unknowns []string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.unknowns = unknowns
|
|
}
|
|
}
|
|
|
|
// ParsedUnknowns returns an argument that sets the values to treat as unknown
|
|
// during partial evaluation.
|
|
func ParsedUnknowns(unknowns []*ast.Term) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.parsedUnknowns = unknowns
|
|
}
|
|
}
|
|
|
|
// DisableInlining adds a set of paths to exclude from partial evaluation inlining.
|
|
func DisableInlining(paths []string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.disableInlining = paths
|
|
}
|
|
}
|
|
|
|
// PartialNamespace returns an argument that sets the namespace to use for
|
|
// partial evaluation results. The namespace must be a valid package path
|
|
// component.
|
|
func PartialNamespace(ns string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.partialNamespace = ns
|
|
}
|
|
}
|
|
|
|
// Module returns an argument that adds a Rego module.
|
|
func Module(filename, input string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.modules = append(r.modules, rawModule{
|
|
filename: filename,
|
|
module: input,
|
|
})
|
|
}
|
|
}
|
|
|
|
// ParsedModule returns an argument that adds a parsed Rego module. If a string
|
|
// module with the same filename name is added, it will override the parsed
|
|
// module.
|
|
func ParsedModule(module *ast.Module) func(*Rego) {
|
|
return func(r *Rego) {
|
|
var filename string
|
|
if module.Package.Location != nil {
|
|
filename = module.Package.Location.File
|
|
} else {
|
|
filename = fmt.Sprintf("module_%p.rego", module)
|
|
}
|
|
r.parsedModules[filename] = module
|
|
}
|
|
}
|
|
|
|
// Load returns an argument that adds a filesystem path to load data
|
|
// and Rego modules from. Any file with a *.rego, *.yaml, or *.json
|
|
// extension will be loaded. The path can be either a directory or file,
|
|
// directories are loaded recursively. The optional ignore string patterns
|
|
// can be used to filter which files are used.
|
|
// The Load option can only be used once.
|
|
// Note: Loading files will require a write transaction on the store.
|
|
func Load(paths []string, filter loader.Filter) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.loadPaths = loadPaths{paths, filter}
|
|
}
|
|
}
|
|
|
|
// LoadBundle returns an argument that adds a filesystem path to load
|
|
// a bundle from. The path can be a compressed bundle file or a directory
|
|
// to be loaded as a bundle.
|
|
// Note: Loading bundles will require a write transaction on the store.
|
|
func LoadBundle(path string) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.bundlePaths = append(r.bundlePaths, path)
|
|
}
|
|
}
|
|
|
|
// ParsedBundle returns an argument that adds a bundle to be loaded.
|
|
func ParsedBundle(name string, b *bundle.Bundle) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.bundles[name] = b
|
|
}
|
|
}
|
|
|
|
// Compiler returns an argument that sets the Rego compiler.
|
|
func Compiler(c *ast.Compiler) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.compiler = c
|
|
}
|
|
}
|
|
|
|
// Store returns an argument that sets the policy engine's data storage layer.
|
|
//
|
|
// If using the Load, LoadBundle, or ParsedBundle options then a transaction
|
|
// must also be provided via the Transaction() option. After loading files
|
|
// or bundles the transaction should be aborted or committed.
|
|
func Store(s storage.Store) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.store = s
|
|
}
|
|
}
|
|
|
|
// Transaction returns an argument that sets the transaction to use for storage
|
|
// layer operations.
|
|
//
|
|
// Requires the store associated with the transaction to be provided via the
|
|
// Store() option. If using Load(), LoadBundle(), or ParsedBundle() options
|
|
// the transaction will likely require write params.
|
|
func Transaction(txn storage.Transaction) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.txn = txn
|
|
}
|
|
}
|
|
|
|
// Metrics returns an argument that sets the metrics collection.
|
|
func Metrics(m metrics.Metrics) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.metrics = m
|
|
}
|
|
}
|
|
|
|
// Instrument returns an argument that enables instrumentation for diagnosing
|
|
// performance issues.
|
|
func Instrument(yes bool) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.instrument = yes
|
|
}
|
|
}
|
|
|
|
// Trace returns an argument that enables tracing on r.
|
|
func Trace(yes bool) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.trace = yes
|
|
}
|
|
}
|
|
|
|
// Tracer returns an argument that adds a query tracer to r.
|
|
func Tracer(t topdown.Tracer) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
if t != nil {
|
|
r.tracers = append(r.tracers, t)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Runtime returns an argument that sets the runtime data to provide to the
|
|
// evaluation engine.
|
|
func Runtime(term *ast.Term) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.runtime = term
|
|
}
|
|
}
|
|
|
|
// PrintTrace is a helper function to write a human-readable version of the
|
|
// trace to the writer w.
|
|
func PrintTrace(w io.Writer, r *Rego) {
|
|
if r == nil || r.tracebuf == nil {
|
|
return
|
|
}
|
|
topdown.PrettyTrace(w, *r.tracebuf)
|
|
}
|
|
|
|
// UnsafeBuiltins sets the built-in functions to treat as unsafe and not allow.
|
|
// This option is ignored for module compilation if the caller supplies the
|
|
// compiler. This option is always honored for query compilation. Provide an
|
|
// empty (non-nil) map to disable checks on queries.
|
|
func UnsafeBuiltins(unsafeBuiltins map[string]struct{}) func(r *Rego) {
|
|
return func(r *Rego) {
|
|
r.unsafeBuiltins = unsafeBuiltins
|
|
}
|
|
}
|
|
|
|
// New returns a new Rego object.
|
|
func New(options ...func(r *Rego)) *Rego {
|
|
|
|
r := &Rego{
|
|
parsedModules: map[string]*ast.Module{},
|
|
capture: map[*ast.Expr]ast.Var{},
|
|
compiledQueries: map[queryType]compiledQuery{},
|
|
builtinDecls: map[string]*ast.Builtin{},
|
|
builtinFuncs: map[string]*topdown.Builtin{},
|
|
bundles: map[string]*bundle.Bundle{},
|
|
}
|
|
|
|
for _, option := range options {
|
|
option(r)
|
|
}
|
|
|
|
if r.compiler == nil {
|
|
r.compiler = ast.NewCompiler().
|
|
WithUnsafeBuiltins(r.unsafeBuiltins).
|
|
WithBuiltins(r.builtinDecls)
|
|
}
|
|
|
|
if r.store == nil {
|
|
r.store = inmem.New()
|
|
r.ownStore = true
|
|
} else {
|
|
r.ownStore = false
|
|
}
|
|
|
|
if r.metrics == nil {
|
|
r.metrics = metrics.New()
|
|
}
|
|
|
|
if r.instrument {
|
|
r.instrumentation = topdown.NewInstrumentation(r.metrics)
|
|
r.compiler.WithMetrics(r.metrics)
|
|
}
|
|
|
|
if r.trace {
|
|
r.tracebuf = topdown.NewBufferTracer()
|
|
r.tracers = append(r.tracers, r.tracebuf)
|
|
}
|
|
|
|
if r.partialNamespace == "" {
|
|
r.partialNamespace = defaultPartialNamespace
|
|
}
|
|
|
|
return r
|
|
}
|
|
|
|
// Eval evaluates this Rego object and returns a ResultSet.
|
|
func (r *Rego) Eval(ctx context.Context) (ResultSet, error) {
|
|
var err error
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pq, err := r.PrepareForEval(ctx)
|
|
if err != nil {
|
|
txnClose(ctx, err) // Ignore error
|
|
return nil, err
|
|
}
|
|
|
|
evalArgs := []EvalOption{
|
|
EvalTransaction(r.txn),
|
|
EvalMetrics(r.metrics),
|
|
EvalInstrument(r.instrument),
|
|
}
|
|
|
|
for _, t := range r.tracers {
|
|
evalArgs = append(evalArgs, EvalTracer(t))
|
|
}
|
|
|
|
rs, err := pq.Eval(ctx, evalArgs...)
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err == nil {
|
|
err = txnErr
|
|
}
|
|
return rs, err
|
|
}
|
|
|
|
// PartialEval has been deprecated and renamed to PartialResult.
|
|
func (r *Rego) PartialEval(ctx context.Context) (PartialResult, error) {
|
|
return r.PartialResult(ctx)
|
|
}
|
|
|
|
// PartialResult partially evaluates this Rego object and returns a PartialResult.
|
|
func (r *Rego) PartialResult(ctx context.Context) (PartialResult, error) {
|
|
var err error
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return PartialResult{}, err
|
|
}
|
|
|
|
pq, err := r.PrepareForEval(ctx, WithPartialEval())
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err != nil {
|
|
return PartialResult{}, err
|
|
}
|
|
if txnErr != nil {
|
|
return PartialResult{}, txnErr
|
|
}
|
|
|
|
pr := PartialResult{
|
|
compiler: pq.r.compiler,
|
|
store: pq.r.store,
|
|
body: pq.r.parsedQuery,
|
|
builtinDecls: pq.r.builtinDecls,
|
|
builtinFuncs: pq.r.builtinFuncs,
|
|
}
|
|
|
|
return pr, nil
|
|
}
|
|
|
|
// Partial runs partial evaluation on r and returns the result.
|
|
func (r *Rego) Partial(ctx context.Context) (*PartialQueries, error) {
|
|
var err error
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pq, err := r.PrepareForPartial(ctx)
|
|
if err != nil {
|
|
txnClose(ctx, err) // Ignore error
|
|
return nil, err
|
|
}
|
|
|
|
evalArgs := []EvalOption{
|
|
EvalTransaction(r.txn),
|
|
EvalMetrics(r.metrics),
|
|
EvalInstrument(r.instrument),
|
|
}
|
|
|
|
for _, t := range r.tracers {
|
|
evalArgs = append(evalArgs, EvalTracer(t))
|
|
}
|
|
|
|
pqs, err := pq.Partial(ctx, evalArgs...)
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err == nil {
|
|
err = txnErr
|
|
}
|
|
return pqs, err
|
|
}
|
|
|
|
// CompileOption defines a function to set options on Compile calls.
|
|
type CompileOption func(*CompileContext)
|
|
|
|
// CompileContext contains options for Compile calls.
|
|
type CompileContext struct {
|
|
partial bool
|
|
}
|
|
|
|
// CompilePartial defines an option to control whether partial evaluation is run
|
|
// before the query is planned and compiled.
|
|
func CompilePartial(yes bool) CompileOption {
|
|
return func(cfg *CompileContext) {
|
|
cfg.partial = yes
|
|
}
|
|
}
|
|
|
|
// Compile returns a compiled policy query.
|
|
func (r *Rego) Compile(ctx context.Context, opts ...CompileOption) (*CompileResult, error) {
|
|
|
|
var cfg CompileContext
|
|
|
|
for _, opt := range opts {
|
|
opt(&cfg)
|
|
}
|
|
|
|
var queries []ast.Body
|
|
var modules []*ast.Module
|
|
|
|
if cfg.partial {
|
|
|
|
pq, err := r.Partial(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if r.dump != nil {
|
|
if len(pq.Queries) != 0 {
|
|
msg := fmt.Sprintf("QUERIES (%d total):", len(pq.Queries))
|
|
fmt.Fprintln(r.dump, msg)
|
|
fmt.Fprintln(r.dump, strings.Repeat("-", len(msg)))
|
|
for i := range pq.Queries {
|
|
fmt.Println(pq.Queries[i])
|
|
}
|
|
fmt.Fprintln(r.dump)
|
|
}
|
|
if len(pq.Support) != 0 {
|
|
msg := fmt.Sprintf("SUPPORT (%d total):", len(pq.Support))
|
|
fmt.Fprintln(r.dump, msg)
|
|
fmt.Fprintln(r.dump, strings.Repeat("-", len(msg)))
|
|
for i := range pq.Support {
|
|
fmt.Println(pq.Support[i])
|
|
}
|
|
fmt.Fprintln(r.dump)
|
|
}
|
|
}
|
|
|
|
queries = pq.Queries
|
|
modules = pq.Support
|
|
|
|
for _, module := range r.compiler.Modules {
|
|
modules = append(modules, module)
|
|
}
|
|
} else {
|
|
var err error
|
|
// If creating a new transacation it should be closed before calling the
|
|
// planner to avoid holding open the transaction longer than needed.
|
|
//
|
|
// TODO(tsandall): in future, planner could make use of store, in which
|
|
// case this will need to change.
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = r.prepare(ctx, compileQueryType, nil)
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if txnErr != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, module := range r.compiler.Modules {
|
|
modules = append(modules, module)
|
|
}
|
|
|
|
queries = []ast.Body{r.compiledQueries[compileQueryType].query}
|
|
}
|
|
|
|
decls := make(map[string]*ast.Builtin, len(r.builtinDecls)+len(ast.BuiltinMap))
|
|
|
|
for k, v := range ast.BuiltinMap {
|
|
decls[k] = v
|
|
}
|
|
|
|
for k, v := range r.builtinDecls {
|
|
decls[k] = v
|
|
}
|
|
|
|
policy, err := planner.New().
|
|
WithQueries(queries).
|
|
WithModules(modules).
|
|
WithRewrittenVars(r.compiledQueries[compileQueryType].compiler.RewrittenVars()).
|
|
WithBuiltinDecls(decls).
|
|
Plan()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if r.dump != nil {
|
|
fmt.Fprintln(r.dump, "PLAN:")
|
|
fmt.Fprintln(r.dump, "-----")
|
|
ir.Pretty(r.dump, policy)
|
|
fmt.Fprintln(r.dump)
|
|
}
|
|
|
|
m, err := wasm.New().WithPolicy(policy).Compile()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var out bytes.Buffer
|
|
|
|
if err := encoding.WriteModule(&out, m); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result := &CompileResult{
|
|
Bytes: out.Bytes(),
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// PrepareOption defines a function to set an option to control
|
|
// the behavior of the Prepare call.
|
|
type PrepareOption func(*PrepareConfig)
|
|
|
|
// PrepareConfig holds settings to control the behavior of the
|
|
// Prepare call.
|
|
type PrepareConfig struct {
|
|
doPartialEval bool
|
|
disableInlining *[]string
|
|
}
|
|
|
|
// WithPartialEval configures an option for PrepareForEval
|
|
// which will have it perform partial evaluation while preparing
|
|
// the query (similar to rego.Rego#PartialResult)
|
|
func WithPartialEval() PrepareOption {
|
|
return func(p *PrepareConfig) {
|
|
p.doPartialEval = true
|
|
}
|
|
}
|
|
|
|
// WithNoInline adds a set of paths to exclude from partial evaluation inlining.
|
|
func WithNoInline(paths []string) PrepareOption {
|
|
return func(p *PrepareConfig) {
|
|
p.disableInlining = &paths
|
|
}
|
|
}
|
|
|
|
// PrepareForEval will parse inputs, modules, and query arguments in preparation
|
|
// of evaluating them.
|
|
func (r *Rego) PrepareForEval(ctx context.Context, opts ...PrepareOption) (PreparedEvalQuery, error) {
|
|
if !r.hasQuery() {
|
|
return PreparedEvalQuery{}, fmt.Errorf("cannot evaluate empty query")
|
|
}
|
|
|
|
pCfg := &PrepareConfig{}
|
|
for _, o := range opts {
|
|
o(pCfg)
|
|
}
|
|
|
|
var err error
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return PreparedEvalQuery{}, err
|
|
}
|
|
|
|
// If the caller wanted to do partial evaluation as part of preparation
|
|
// do it now and use the new Rego object.
|
|
if pCfg.doPartialEval {
|
|
|
|
pr, err := r.partialResult(ctx, pCfg)
|
|
if err != nil {
|
|
txnClose(ctx, err) // Ignore error
|
|
return PreparedEvalQuery{}, err
|
|
}
|
|
|
|
// Prepare the new query using the result of partial evaluation
|
|
pq, err := pr.Rego(Transaction(r.txn)).PrepareForEval(ctx)
|
|
txnErr := txnClose(ctx, err)
|
|
if err != nil {
|
|
return pq, err
|
|
}
|
|
return pq, txnErr
|
|
}
|
|
|
|
err = r.prepare(ctx, evalQueryType, []extraStage{
|
|
{
|
|
after: "ResolveRefs",
|
|
stage: ast.QueryCompilerStageDefinition{
|
|
Name: "RewriteToCaptureValue",
|
|
MetricName: "query_compile_stage_rewrite_to_capture_value",
|
|
Stage: r.rewriteQueryToCaptureValue,
|
|
},
|
|
},
|
|
})
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err != nil {
|
|
return PreparedEvalQuery{}, err
|
|
}
|
|
if txnErr != nil {
|
|
return PreparedEvalQuery{}, txnErr
|
|
}
|
|
|
|
return PreparedEvalQuery{preparedQuery{r, pCfg}}, err
|
|
}
|
|
|
|
// PrepareForPartial will parse inputs, modules, and query arguments in preparation
|
|
// of partially evaluating them.
|
|
func (r *Rego) PrepareForPartial(ctx context.Context, opts ...PrepareOption) (PreparedPartialQuery, error) {
|
|
if !r.hasQuery() {
|
|
return PreparedPartialQuery{}, fmt.Errorf("cannot evaluate empty query")
|
|
}
|
|
|
|
pCfg := &PrepareConfig{}
|
|
for _, o := range opts {
|
|
o(pCfg)
|
|
}
|
|
|
|
var err error
|
|
var txnClose transactionCloser
|
|
r.txn, txnClose, err = r.getTxn(ctx)
|
|
if err != nil {
|
|
return PreparedPartialQuery{}, err
|
|
}
|
|
|
|
err = r.prepare(ctx, partialQueryType, []extraStage{
|
|
{
|
|
after: "CheckSafety",
|
|
stage: ast.QueryCompilerStageDefinition{
|
|
Name: "RewriteEquals",
|
|
MetricName: "query_compile_stage_rewrite_equals",
|
|
Stage: r.rewriteEqualsForPartialQueryCompile,
|
|
},
|
|
},
|
|
})
|
|
txnErr := txnClose(ctx, err) // Always call closer
|
|
if err != nil {
|
|
return PreparedPartialQuery{}, err
|
|
}
|
|
if txnErr != nil {
|
|
return PreparedPartialQuery{}, txnErr
|
|
}
|
|
return PreparedPartialQuery{preparedQuery{r, pCfg}}, err
|
|
}
|
|
|
|
func (r *Rego) prepare(ctx context.Context, qType queryType, extras []extraStage) error {
|
|
var err error
|
|
|
|
r.parsedInput, err = r.parseInput()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = r.loadFiles(ctx, r.txn, r.metrics)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = r.loadBundles(ctx, r.txn, r.metrics)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = r.parseModules(ctx, r.txn, r.metrics)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Compile the modules *before* the query, else functions
|
|
// defined in the module won't be found...
|
|
err = r.compileModules(ctx, r.txn, r.metrics)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
r.parsedQuery, err = r.parseQuery(r.metrics)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = r.compileAndCacheQuery(qType, r.parsedQuery, r.metrics, extras)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) parseModules(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error {
|
|
if len(r.modules) == 0 {
|
|
return nil
|
|
}
|
|
|
|
m.Timer(metrics.RegoModuleParse).Start()
|
|
defer m.Timer(metrics.RegoModuleParse).Stop()
|
|
var errs Errors
|
|
|
|
// Parse any modules in the are saved to the store, but only if
|
|
// another compile step is going to occur (ie. we have parsed modules
|
|
// that need to be compiled).
|
|
ids, err := r.store.ListPolicies(ctx, txn)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, id := range ids {
|
|
// if it is already on the compiler we're using
|
|
// then don't bother to re-parse it from source
|
|
if _, haveMod := r.compiler.Modules[id]; haveMod {
|
|
continue
|
|
}
|
|
|
|
bs, err := r.store.GetPolicy(ctx, txn, id)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
parsed, err := ast.ParseModule(id, string(bs))
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
|
|
r.parsedModules[id] = parsed
|
|
}
|
|
|
|
// Parse any passed in as arguments to the Rego object
|
|
for _, module := range r.modules {
|
|
p, err := module.Parse()
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
r.parsedModules[module.filename] = p
|
|
}
|
|
|
|
if len(errs) > 0 {
|
|
return errs
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) loadFiles(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error {
|
|
if len(r.loadPaths.paths) == 0 {
|
|
return nil
|
|
}
|
|
|
|
m.Timer(metrics.RegoLoadFiles).Start()
|
|
defer m.Timer(metrics.RegoLoadFiles).Stop()
|
|
|
|
result, err := loader.NewFileLoader().WithMetrics(m).Filtered(r.loadPaths.paths, r.loadPaths.filter)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for name, mod := range result.Modules {
|
|
r.parsedModules[name] = mod.Parsed
|
|
}
|
|
|
|
if len(result.Documents) > 0 {
|
|
err = r.store.Write(ctx, txn, storage.AddOp, storage.Path{}, result.Documents)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) loadBundles(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error {
|
|
if len(r.bundlePaths) == 0 {
|
|
return nil
|
|
}
|
|
|
|
m.Timer(metrics.RegoLoadBundles).Start()
|
|
defer m.Timer(metrics.RegoLoadBundles).Stop()
|
|
|
|
for _, path := range r.bundlePaths {
|
|
bndl, err := loader.NewFileLoader().WithMetrics(m).AsBundle(path)
|
|
if err != nil {
|
|
return fmt.Errorf("loading error: %s", err)
|
|
}
|
|
r.bundles[path] = bndl
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) parseInput() (ast.Value, error) {
|
|
if r.parsedInput != nil {
|
|
return r.parsedInput, nil
|
|
}
|
|
return r.parseRawInput(r.rawInput, r.metrics)
|
|
}
|
|
|
|
func (r *Rego) parseRawInput(rawInput *interface{}, m metrics.Metrics) (ast.Value, error) {
|
|
var input ast.Value
|
|
|
|
if rawInput == nil {
|
|
return input, nil
|
|
}
|
|
|
|
m.Timer(metrics.RegoInputParse).Start()
|
|
defer m.Timer(metrics.RegoInputParse).Stop()
|
|
|
|
rawPtr := util.Reference(rawInput)
|
|
|
|
// roundtrip through json: this turns slices (e.g. []string, []bool) into
|
|
// []interface{}, the only array type ast.InterfaceToValue can work with
|
|
if err := util.RoundTrip(rawPtr); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return ast.InterfaceToValue(*rawPtr)
|
|
}
|
|
|
|
func (r *Rego) parseQuery(m metrics.Metrics) (ast.Body, error) {
|
|
if r.parsedQuery != nil {
|
|
return r.parsedQuery, nil
|
|
}
|
|
|
|
m.Timer(metrics.RegoQueryParse).Start()
|
|
defer m.Timer(metrics.RegoQueryParse).Stop()
|
|
|
|
return ast.ParseBody(r.query)
|
|
}
|
|
|
|
func (r *Rego) compileModules(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error {
|
|
|
|
// Only compile again if there are new modules.
|
|
if len(r.bundles) > 0 || len(r.parsedModules) > 0 {
|
|
|
|
// The bundle.Activate call will activate any bundles passed in
|
|
// (ie compile + handle data store changes), and include any of
|
|
// the additional modules passed in. If no bundles are provided
|
|
// it will only compile the passed in modules.
|
|
// Use this as the single-point of compiling everything only a
|
|
// single time.
|
|
opts := &bundle.ActivateOpts{
|
|
Ctx: ctx,
|
|
Store: r.store,
|
|
Txn: txn,
|
|
Compiler: r.compiler.WithPathConflictsCheck(storage.NonEmpty(ctx, r.store, txn)),
|
|
Metrics: m,
|
|
Bundles: r.bundles,
|
|
ExtraModules: r.parsedModules,
|
|
}
|
|
err := bundle.Activate(opts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) compileAndCacheQuery(qType queryType, query ast.Body, m metrics.Metrics, extras []extraStage) error {
|
|
m.Timer(metrics.RegoQueryCompile).Start()
|
|
defer m.Timer(metrics.RegoQueryCompile).Stop()
|
|
|
|
cachedQuery, ok := r.compiledQueries[qType]
|
|
if ok && cachedQuery.query != nil && cachedQuery.compiler != nil {
|
|
return nil
|
|
}
|
|
|
|
qc, compiled, err := r.compileQuery(query, m, extras)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// cache the query for future use
|
|
r.compiledQueries[qType] = compiledQuery{
|
|
query: compiled,
|
|
compiler: qc,
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Rego) compileQuery(query ast.Body, m metrics.Metrics, extras []extraStage) (ast.QueryCompiler, ast.Body, error) {
|
|
var pkg *ast.Package
|
|
|
|
if r.pkg != "" {
|
|
var err error
|
|
pkg, err = ast.ParsePackage(fmt.Sprintf("package %v", r.pkg))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
} else {
|
|
pkg = r.parsedPackage
|
|
}
|
|
|
|
imports := r.parsedImports
|
|
|
|
if len(r.imports) > 0 {
|
|
s := make([]string, len(r.imports))
|
|
for i := range r.imports {
|
|
s[i] = fmt.Sprintf("import %v", r.imports[i])
|
|
}
|
|
parsed, err := ast.ParseImports(strings.Join(s, "\n"))
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
imports = append(imports, parsed...)
|
|
}
|
|
|
|
qctx := ast.NewQueryContext().
|
|
WithPackage(pkg).
|
|
WithImports(imports)
|
|
|
|
qc := r.compiler.QueryCompiler().
|
|
WithContext(qctx).
|
|
WithUnsafeBuiltins(r.unsafeBuiltins)
|
|
|
|
for _, extra := range extras {
|
|
qc = qc.WithStageAfter(extra.after, extra.stage)
|
|
}
|
|
|
|
compiled, err := qc.Compile(query)
|
|
|
|
return qc, compiled, err
|
|
|
|
}
|
|
|
|
func (r *Rego) eval(ctx context.Context, ectx *EvalContext) (ResultSet, error) {
|
|
|
|
q := topdown.NewQuery(ectx.compiledQuery.query).
|
|
WithQueryCompiler(ectx.compiledQuery.compiler).
|
|
WithCompiler(r.compiler).
|
|
WithStore(r.store).
|
|
WithTransaction(ectx.txn).
|
|
WithBuiltins(r.builtinFuncs).
|
|
WithMetrics(ectx.metrics).
|
|
WithInstrumentation(ectx.instrumentation).
|
|
WithRuntime(r.runtime).
|
|
WithIndexing(ectx.indexing)
|
|
|
|
for i := range ectx.tracers {
|
|
q = q.WithTracer(ectx.tracers[i])
|
|
}
|
|
|
|
if ectx.parsedInput != nil {
|
|
q = q.WithInput(ast.NewTerm(ectx.parsedInput))
|
|
}
|
|
|
|
// Cancel query if context is cancelled or deadline is reached.
|
|
c := topdown.NewCancel()
|
|
q = q.WithCancel(c)
|
|
exit := make(chan struct{})
|
|
defer close(exit)
|
|
go waitForDone(ctx, exit, func() {
|
|
c.Cancel()
|
|
})
|
|
|
|
rewritten := ectx.compiledQuery.compiler.RewrittenVars()
|
|
var rs ResultSet
|
|
err := q.Iter(ctx, func(qr topdown.QueryResult) error {
|
|
result := newResult()
|
|
for k := range qr {
|
|
v, err := ast.JSON(qr[k].Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if rw, ok := rewritten[k]; ok {
|
|
k = rw
|
|
}
|
|
if isTermVar(k) || k.IsGenerated() || k.IsWildcard() {
|
|
continue
|
|
}
|
|
result.Bindings[string(k)] = v
|
|
}
|
|
for _, expr := range ectx.compiledQuery.query {
|
|
if expr.Generated {
|
|
continue
|
|
}
|
|
if k, ok := r.capture[expr]; ok {
|
|
v, err := ast.JSON(qr[k].Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
result.Expressions = append(result.Expressions, newExpressionValue(expr, v))
|
|
} else {
|
|
result.Expressions = append(result.Expressions, newExpressionValue(expr, true))
|
|
}
|
|
}
|
|
rs = append(rs, result)
|
|
return nil
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(rs) == 0 {
|
|
return nil, nil
|
|
}
|
|
|
|
return rs, nil
|
|
}
|
|
|
|
func (r *Rego) partialResult(ctx context.Context, pCfg *PrepareConfig) (PartialResult, error) {
|
|
|
|
err := r.prepare(ctx, partialResultQueryType, []extraStage{
|
|
{
|
|
after: "ResolveRefs",
|
|
stage: ast.QueryCompilerStageDefinition{
|
|
Name: "RewriteForPartialEval",
|
|
MetricName: "query_compile_stage_rewrite_for_partial_eval",
|
|
Stage: r.rewriteQueryForPartialEval,
|
|
},
|
|
},
|
|
})
|
|
if err != nil {
|
|
return PartialResult{}, err
|
|
}
|
|
|
|
ectx := &EvalContext{
|
|
parsedInput: r.parsedInput,
|
|
metrics: r.metrics,
|
|
txn: r.txn,
|
|
partialNamespace: r.partialNamespace,
|
|
tracers: r.tracers,
|
|
compiledQuery: r.compiledQueries[partialResultQueryType],
|
|
instrumentation: r.instrumentation,
|
|
indexing: true,
|
|
}
|
|
|
|
disableInlining := r.disableInlining
|
|
|
|
if pCfg.disableInlining != nil {
|
|
disableInlining = *pCfg.disableInlining
|
|
}
|
|
|
|
ectx.disableInlining, err = parseStringsToRefs(disableInlining)
|
|
if err != nil {
|
|
return PartialResult{}, err
|
|
}
|
|
|
|
pq, err := r.partial(ctx, ectx)
|
|
if err != nil {
|
|
return PartialResult{}, err
|
|
}
|
|
|
|
// Construct module for queries.
|
|
module := ast.MustParseModule("package " + ectx.partialNamespace)
|
|
module.Rules = make([]*ast.Rule, len(pq.Queries))
|
|
for i, body := range pq.Queries {
|
|
module.Rules[i] = &ast.Rule{
|
|
Head: ast.NewHead(ast.Var("__result__"), nil, ast.Wildcard),
|
|
Body: body,
|
|
Module: module,
|
|
}
|
|
}
|
|
|
|
// Update compiler with partial evaluation output.
|
|
r.compiler.Modules["__partialresult__"] = module
|
|
for i, module := range pq.Support {
|
|
r.compiler.Modules[fmt.Sprintf("__partialsupport%d__", i)] = module
|
|
}
|
|
|
|
r.metrics.Timer(metrics.RegoModuleCompile).Start()
|
|
r.compiler.Compile(r.compiler.Modules)
|
|
r.metrics.Timer(metrics.RegoModuleCompile).Stop()
|
|
|
|
if r.compiler.Failed() {
|
|
return PartialResult{}, r.compiler.Errors
|
|
}
|
|
|
|
result := PartialResult{
|
|
compiler: r.compiler,
|
|
store: r.store,
|
|
body: ast.MustParseBody(fmt.Sprintf("data.%v.__result__", ectx.partialNamespace)),
|
|
builtinDecls: r.builtinDecls,
|
|
builtinFuncs: r.builtinFuncs,
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func (r *Rego) partial(ctx context.Context, ectx *EvalContext) (*PartialQueries, error) {
|
|
|
|
var unknowns []*ast.Term
|
|
|
|
if ectx.parsedUnknowns != nil {
|
|
unknowns = ectx.parsedUnknowns
|
|
} else if ectx.unknowns != nil {
|
|
unknowns = make([]*ast.Term, len(ectx.unknowns))
|
|
for i := range ectx.unknowns {
|
|
var err error
|
|
unknowns[i], err = ast.ParseTerm(ectx.unknowns[i])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
} else {
|
|
// Use input document as unknown if caller has not specified any.
|
|
unknowns = []*ast.Term{ast.NewTerm(ast.InputRootRef)}
|
|
}
|
|
|
|
// Check partial namespace to ensure it's valid.
|
|
if term, err := ast.ParseTerm(ectx.partialNamespace); err != nil {
|
|
return nil, err
|
|
} else if _, ok := term.Value.(ast.Var); !ok {
|
|
return nil, fmt.Errorf("bad partial namespace")
|
|
}
|
|
|
|
q := topdown.NewQuery(ectx.compiledQuery.query).
|
|
WithQueryCompiler(ectx.compiledQuery.compiler).
|
|
WithCompiler(r.compiler).
|
|
WithStore(r.store).
|
|
WithTransaction(ectx.txn).
|
|
WithBuiltins(r.builtinFuncs).
|
|
WithMetrics(ectx.metrics).
|
|
WithInstrumentation(ectx.instrumentation).
|
|
WithUnknowns(unknowns).
|
|
WithDisableInlining(ectx.disableInlining).
|
|
WithRuntime(r.runtime).
|
|
WithIndexing(ectx.indexing)
|
|
|
|
for i := range ectx.tracers {
|
|
q = q.WithTracer(ectx.tracers[i])
|
|
}
|
|
|
|
if ectx.parsedInput != nil {
|
|
q = q.WithInput(ast.NewTerm(ectx.parsedInput))
|
|
}
|
|
|
|
// Cancel query if context is cancelled or deadline is reached.
|
|
c := topdown.NewCancel()
|
|
q = q.WithCancel(c)
|
|
exit := make(chan struct{})
|
|
defer close(exit)
|
|
go waitForDone(ctx, exit, func() {
|
|
c.Cancel()
|
|
})
|
|
|
|
queries, support, err := q.PartialRun(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pq := &PartialQueries{
|
|
Queries: queries,
|
|
Support: support,
|
|
}
|
|
|
|
return pq, nil
|
|
}
|
|
|
|
func (r *Rego) rewriteQueryToCaptureValue(qc ast.QueryCompiler, query ast.Body) (ast.Body, error) {
|
|
|
|
checkCapture := iteration(query) || len(query) > 1
|
|
|
|
for _, expr := range query {
|
|
|
|
if expr.Negated {
|
|
continue
|
|
}
|
|
|
|
if expr.IsAssignment() || expr.IsEquality() {
|
|
continue
|
|
}
|
|
|
|
var capture *ast.Term
|
|
|
|
// If the expression can be evaluated as a function, rewrite it to
|
|
// capture the return value. E.g., neq(1,2) becomes neq(1,2,x) but
|
|
// plus(1,2,x) does not get rewritten.
|
|
switch terms := expr.Terms.(type) {
|
|
case *ast.Term:
|
|
capture = r.generateTermVar()
|
|
expr.Terms = ast.Equality.Expr(terms, capture).Terms
|
|
r.capture[expr] = capture.Value.(ast.Var)
|
|
case []*ast.Term:
|
|
if r.compiler.GetArity(expr.Operator()) == len(terms)-1 {
|
|
capture = r.generateTermVar()
|
|
expr.Terms = append(terms, capture)
|
|
r.capture[expr] = capture.Value.(ast.Var)
|
|
}
|
|
}
|
|
|
|
if capture != nil && checkCapture {
|
|
cpy := expr.Copy()
|
|
cpy.Terms = capture
|
|
cpy.Generated = true
|
|
cpy.With = nil
|
|
query.Append(cpy)
|
|
}
|
|
}
|
|
|
|
return query, nil
|
|
}
|
|
|
|
func (r *Rego) rewriteQueryForPartialEval(_ ast.QueryCompiler, query ast.Body) (ast.Body, error) {
|
|
if len(query) != 1 {
|
|
return nil, fmt.Errorf("partial evaluation requires single ref (not multiple expressions)")
|
|
}
|
|
|
|
term, ok := query[0].Terms.(*ast.Term)
|
|
if !ok {
|
|
return nil, fmt.Errorf("partial evaluation requires ref (not expression)")
|
|
}
|
|
|
|
ref, ok := term.Value.(ast.Ref)
|
|
if !ok {
|
|
return nil, fmt.Errorf("partial evaluation requires ref (not %v)", ast.TypeName(term.Value))
|
|
}
|
|
|
|
if !ref.IsGround() {
|
|
return nil, fmt.Errorf("partial evaluation requires ground ref")
|
|
}
|
|
|
|
return ast.NewBody(ast.Equality.Expr(ast.Wildcard, term)), nil
|
|
}
|
|
|
|
// rewriteEqualsForPartialQueryCompile will rewrite == to = in queries. Normally
|
|
// this wouldn't be done, except for handling queries with the `Partial` API
|
|
// where rewriting them can substantially simplify the result, and it is unlikely
|
|
// that the caller would need expression values.
|
|
func (r *Rego) rewriteEqualsForPartialQueryCompile(_ ast.QueryCompiler, query ast.Body) (ast.Body, error) {
|
|
doubleEq := ast.Equal.Ref()
|
|
unifyOp := ast.Equality.Ref()
|
|
ast.WalkExprs(query, func(x *ast.Expr) bool {
|
|
if x.IsCall() {
|
|
operator := x.Operator()
|
|
if operator.Equal(doubleEq) && len(x.Operands()) == 2 {
|
|
x.SetOperator(ast.NewTerm(unifyOp))
|
|
}
|
|
}
|
|
return false
|
|
})
|
|
return query, nil
|
|
}
|
|
|
|
func (r *Rego) generateTermVar() *ast.Term {
|
|
r.termVarID++
|
|
return ast.VarTerm(ast.WildcardPrefix + fmt.Sprintf("term%v", r.termVarID))
|
|
}
|
|
|
|
func (r Rego) hasQuery() bool {
|
|
return len(r.query) != 0 || len(r.parsedQuery) != 0
|
|
}
|
|
|
|
type transactionCloser func(ctx context.Context, err error) error
|
|
|
|
// getTxn will conditionally create a read or write transaction suitable for
|
|
// the configured Rego object. The returned function should be used to close the txn
|
|
// regardless of status.
|
|
func (r *Rego) getTxn(ctx context.Context) (storage.Transaction, transactionCloser, error) {
|
|
|
|
noopCloser := func(ctx context.Context, err error) error {
|
|
return nil // no-op default
|
|
}
|
|
|
|
if r.txn != nil {
|
|
// Externally provided txn
|
|
return r.txn, noopCloser, nil
|
|
}
|
|
|
|
// Create a new transaction..
|
|
params := storage.TransactionParams{}
|
|
|
|
// Bundles and data paths may require writing data files or manifests to storage
|
|
if len(r.bundles) > 0 || len(r.bundlePaths) > 0 || len(r.loadPaths.paths) > 0 {
|
|
|
|
// If we were given a store we will *not* write to it, only do that on one
|
|
// which was created automatically on behalf of the user.
|
|
if !r.ownStore {
|
|
return nil, noopCloser, errors.New("unable to start write transaction when store was provided")
|
|
}
|
|
|
|
params.Write = true
|
|
}
|
|
|
|
txn, err := r.store.NewTransaction(ctx, params)
|
|
if err != nil {
|
|
return nil, noopCloser, err
|
|
}
|
|
|
|
// Setup a closer function that will abort or commit as needed.
|
|
closer := func(ctx context.Context, txnErr error) error {
|
|
var err error
|
|
|
|
if txnErr == nil && params.Write {
|
|
err = r.store.Commit(ctx, txn)
|
|
} else {
|
|
r.store.Abort(ctx, txn)
|
|
}
|
|
|
|
// Clear the auto created transaction now that it is closed.
|
|
r.txn = nil
|
|
|
|
return err
|
|
}
|
|
|
|
return txn, closer, nil
|
|
}
|
|
|
|
func isTermVar(v ast.Var) bool {
|
|
return strings.HasPrefix(string(v), ast.WildcardPrefix+"term")
|
|
}
|
|
|
|
func waitForDone(ctx context.Context, exit chan struct{}, f func()) {
|
|
select {
|
|
case <-exit:
|
|
return
|
|
case <-ctx.Done():
|
|
f()
|
|
return
|
|
}
|
|
}
|
|
|
|
type rawModule struct {
|
|
filename string
|
|
module string
|
|
}
|
|
|
|
func (m rawModule) Parse() (*ast.Module, error) {
|
|
return ast.ParseModule(m.filename, m.module)
|
|
}
|
|
|
|
type extraStage struct {
|
|
after string
|
|
stage ast.QueryCompilerStageDefinition
|
|
}
|
|
|
|
func iteration(x interface{}) bool {
|
|
|
|
var stopped bool
|
|
|
|
vis := ast.NewGenericVisitor(func(x interface{}) bool {
|
|
switch x := x.(type) {
|
|
case *ast.Term:
|
|
if ast.IsComprehension(x.Value) {
|
|
return true
|
|
}
|
|
case ast.Ref:
|
|
if !stopped {
|
|
if bi := ast.BuiltinMap[x.String()]; bi != nil {
|
|
if bi.Relation {
|
|
stopped = true
|
|
return stopped
|
|
}
|
|
}
|
|
for i := 1; i < len(x); i++ {
|
|
if _, ok := x[i].Value.(ast.Var); ok {
|
|
stopped = true
|
|
return stopped
|
|
}
|
|
}
|
|
}
|
|
return stopped
|
|
}
|
|
return stopped
|
|
})
|
|
|
|
vis.Walk(x)
|
|
|
|
return stopped
|
|
}
|
|
|
|
func parseStringsToRefs(s []string) ([]ast.Ref, error) {
|
|
|
|
refs := make([]ast.Ref, len(s))
|
|
for i := range refs {
|
|
var err error
|
|
refs[i], err = ast.ParseRef(s[i])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return refs, nil
|
|
}
|
|
|
|
// helper function to finish a built-in function call. If an error occured,
|
|
// wrap the error and return it. Otherwise, invoke the iterator if the result
|
|
// was defined.
|
|
func finishFunction(name string, bctx topdown.BuiltinContext, result *ast.Term, err error, iter func(*ast.Term) error) error {
|
|
if err != nil {
|
|
return &topdown.Error{
|
|
Code: topdown.BuiltinErr,
|
|
Message: fmt.Sprintf("%v: %v", name, err.Error()),
|
|
Location: bctx.Location,
|
|
}
|
|
}
|
|
if result == nil {
|
|
return nil
|
|
}
|
|
return iter(result)
|
|
}
|
|
|
|
// helper function to return an option that sets a custom built-in function.
|
|
func newFunction(decl *Function, f topdown.BuiltinFunc) func(*Rego) {
|
|
return func(r *Rego) {
|
|
r.builtinDecls[decl.Name] = &ast.Builtin{
|
|
Name: decl.Name,
|
|
Decl: decl.Decl,
|
|
}
|
|
r.builtinFuncs[decl.Name] = &topdown.Builtin{
|
|
Decl: r.builtinDecls[decl.Name],
|
|
Func: f,
|
|
}
|
|
}
|
|
}
|