Skip to content

Commit

Permalink
Facilitate OPA decision correlation with business flows (#3041)
Browse files Browse the repository at this point in the history
* feature: OPA decision correlation with business flows

Signed-off-by: Magnus Jungsbluth <[email protected]>
Signed-off-by: Janardhan Sharma <[email protected]>
  • Loading branch information
JanardhanSharma authored Jun 24, 2024
1 parent 8e2c2cc commit 2a623a9
Show file tree
Hide file tree
Showing 4 changed files with 106 additions and 15 deletions.
9 changes: 9 additions & 0 deletions docs/tutorials/auth.md
Original file line number Diff line number Diff line change
Expand Up @@ -466,6 +466,15 @@ The second argument is parsed as YAML, cannot be nested and values need to be st
In Rego this can be used like this `input.attributes.contextExtensions["com.mycompany.myprop"] == "my value"`
### Decision ID in Policies
Each evaluation yields a distinct decision, identifiable by its unique decision ID.
This decision ID can be located within the input at:
`input.attributes.metadataContext.filterMetadata.open_policy_agent.decision_id`
Typical use cases are either propagation of the decision ID to downstream systems or returning it as part of the response. As an example this can allow to trouble shoot deny requests by looking up details using the full decision in a control plane.
### Quick Start Rego Playground
A quick way without setting up Backend APIs is to use the [Rego Playground](https://play.openpolicyagent.org/).
Expand Down
34 changes: 32 additions & 2 deletions filters/openpolicyagent/evaluation.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,7 @@ package openpolicyagent
import (
"context"
"fmt"
"time"

ext_authz_v3_core "github.com/envoyproxy/go-control-plane/envoy/config/core/v3"
ext_authz_v3 "github.com/envoyproxy/go-control-plane/envoy/service/auth/v3"
"github.com/open-policy-agent/opa-envoy-plugin/envoyauth"
"github.com/open-policy-agent/opa-envoy-plugin/opa/decisionlog"
Expand All @@ -13,6 +12,8 @@ import (
"github.com/open-policy-agent/opa/server"
"github.com/open-policy-agent/opa/topdown"
"github.com/opentracing/opentracing-go"
pbstruct "google.golang.org/protobuf/types/known/structpb"
"time"
)

func (opa *OpenPolicyAgentInstance) Eval(ctx context.Context, req *ext_authz_v3.CheckRequest) (*envoyauth.EvalResult, error) {
Expand All @@ -23,6 +24,12 @@ func (opa *OpenPolicyAgentInstance) Eval(ctx context.Context, req *ext_authz_v3.
return nil, err
}

err = setDecisionIdInRequest(req, decisionId)
if err != nil {
opa.Logger().WithFields(map[string]interface{}{"err": err}).Error("Unable to set decision ID in Request.")
return nil, err
}

result, stopeval, err := envoyauth.NewEvalResult(withDecisionID(decisionId))
if err != nil {
opa.Logger().WithFields(map[string]interface{}{"err": err}).Error("Unable to generate new result with decision ID.")
Expand Down Expand Up @@ -71,6 +78,29 @@ func (opa *OpenPolicyAgentInstance) Eval(ctx context.Context, req *ext_authz_v3.
return result, nil
}

func setDecisionIdInRequest(req *ext_authz_v3.CheckRequest, decisionId string) error {
if req.Attributes.MetadataContext == nil {
req.Attributes.MetadataContext = &ext_authz_v3_core.Metadata{
FilterMetadata: map[string]*pbstruct.Struct{},
}
}

filterMeta, err := formOpenPolicyAgentMetaDataObject(decisionId)
if err != nil {
return err
}
req.Attributes.MetadataContext.FilterMetadata["open_policy_agent"] = filterMeta
return nil
}

func formOpenPolicyAgentMetaDataObject(decisionId string) (*pbstruct.Struct, error) {

innerFields := make(map[string]interface{})
innerFields["decision_id"] = decisionId

return pbstruct.NewStruct(innerFields)
}

func (opa *OpenPolicyAgentInstance) logDecision(ctx context.Context, input interface{}, result *envoyauth.EvalResult, err error) error {
info := &server.Info{
Timestamp: time.Now(),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -255,6 +255,21 @@ func TestAuthorizeRequestFilter(t *testing.T) {
backendHeaders: make(http.Header),
removeHeaders: make(http.Header),
},
{
msg: "Decision id in request header",
filterName: "opaAuthorizeRequest",
bundleName: "somebundle.tar.gz",
regoQuery: "envoy/authz/allow_object_decision_id_in_header",
requestMethod: "POST",
body: `{ "target_id" : "123456" }`,
requestHeaders: map[string][]string{"content-type": {"application/json"}},
requestPath: "/allow/structured",
expectedStatus: http.StatusOK,
expectedBody: "Welcome!",
expectedHeaders: map[string][]string{"Decision-Id": {"some-random-decision-id-generated-during-evaluation"}},
backendHeaders: make(http.Header),
removeHeaders: make(http.Header),
},
{
msg: "Invalid UTF-8 in Path",
filterName: "opaAuthorizeRequest",
Expand Down Expand Up @@ -345,8 +360,21 @@ func TestAuthorizeRequestFilter(t *testing.T) {
allow_body {
input.parsed_body.target_id == "123456"
}
`,
}
decision_id := input.attributes.metadataContext.filterMetadata.open_policy_agent.decision_id
allow_object_decision_id_in_header = response {
input.parsed_path = ["allow", "structured"]
decision_id
response := {
"allowed": true,
"response_headers_to_add": {
"decision-id": decision_id
}
}
}
`,
}),
)

Expand Down Expand Up @@ -374,10 +402,24 @@ func TestAuthorizeRequestFilter(t *testing.T) {
}
}`, opaControlPlane.URL(), ti.regoQuery))

envoyMetaDataConfig := []byte(`{
"filter_metadata": {
"envoy.filters.http.header_to_metadata": {
"policy_type": "ingress"
}
}
}`)

opts := make([]func(*openpolicyagent.OpenPolicyAgentInstanceConfig) error, 0)
opts = append(opts,
openpolicyagent.WithConfigTemplate(config),
openpolicyagent.WithEnvoyMetadataBytes(envoyMetaDataConfig))

opaFactory := openpolicyagent.NewOpenPolicyAgentRegistry(openpolicyagent.WithTracer(&tracingtest.Tracer{}))
ftSpec := NewOpaAuthorizeRequestSpec(opaFactory, openpolicyagent.WithConfigTemplate(config))
ftSpec := NewOpaAuthorizeRequestSpec(opaFactory, opts...)

fr.Register(ftSpec)
ftSpec = NewOpaAuthorizeRequestWithBodySpec(opaFactory, openpolicyagent.WithConfigTemplate(config))
ftSpec = NewOpaAuthorizeRequestWithBodySpec(opaFactory, opts...)
fr.Register(ftSpec)

r := eskip.MustParse(fmt.Sprintf(`* -> %s("%s", "%s") %s -> "%s"`, ti.filterName, ti.bundleName, ti.contextExtensions, ti.extraeskip, clientServer.URL))
Expand Down Expand Up @@ -435,8 +477,12 @@ func isHeadersPresent(t *testing.T, expectedHeaders http.Header, headers http.He
if !headerFound {
return false
}

assert.ElementsMatch(t, expectedValues, actualValues)
// since decision id is randomly generated we are just checking for not nil
if headerName == "Decision-Id" {
assert.NotNil(t, actualValues)
} else {
assert.ElementsMatch(t, expectedValues, actualValues)
}
}
return true
}
Expand Down
20 changes: 13 additions & 7 deletions filters/openpolicyagent/openpolicyagent_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ import (
"github.com/zalando/skipper/routing"
"github.com/zalando/skipper/tracing/tracingtest"
"google.golang.org/protobuf/encoding/protojson"
_struct "google.golang.org/protobuf/types/known/structpb"
pbstruct "google.golang.org/protobuf/types/known/structpb"
)

type MockOpenPolicyAgentFilter struct {
Expand Down Expand Up @@ -63,22 +63,22 @@ func TestInterpolateTemplate(t *testing.T) {

func TestLoadEnvoyMetadata(t *testing.T) {
cfg := &OpenPolicyAgentInstanceConfig{}
WithEnvoyMetadataBytes([]byte(`
_ = WithEnvoyMetadataBytes([]byte(`
{
"filter_metadata": {
"envoy.filters.http.header_to_metadata": {
"policy_type": "ingress"
}
},
}
}
`))(cfg)

expectedBytes, err := protojson.Marshal(&ext_authz_v3_core.Metadata{
FilterMetadata: map[string]*_struct.Struct{
FilterMetadata: map[string]*pbstruct.Struct{
"envoy.filters.http.header_to_metadata": {
Fields: map[string]*_struct.Value{
Fields: map[string]*pbstruct.Value{
"policy_type": {
Kind: &_struct.Value_StringValue{StringValue: "ingress"},
Kind: &pbstruct.Value_StringValue{StringValue: "ingress"},
},
},
},
Expand Down Expand Up @@ -411,7 +411,13 @@ func TestEval(t *testing.T) {
span := tracer.StartSpan("open-policy-agent")
ctx := opentracing.ContextWithSpan(context.Background(), span)

result, err := inst.Eval(ctx, &authv3.CheckRequest{})
result, err := inst.Eval(ctx, &authv3.CheckRequest{
Attributes: &authv3.AttributeContext{
Request: nil,
ContextExtensions: nil,
MetadataContext: nil,
},
})
assert.NoError(t, err)

allowed, err := result.IsAllowed()
Expand Down

0 comments on commit 2a623a9

Please sign in to comment.