When integrating with OPA there are two interfaces to consider:

    • Evaluation: OPA’s interface for asking for policy decisions. Integrating OPA is primarily focused on integrating an application, service, or tool with OPA’s policy evaluation interface. This integration results in policy decisions being decoupled from that application, service, or tool.
    • Management: OPA’s interface for deploying policies, understanding status, uploading logs, and so on. This integration is typically the same across all OPA instances, regardless what software the evaluation interface is integrated with. Distributing policy, retrieving status, and storing logs in the same way across all OPAs provides a unified management plane for policy across many different software systems.

    This page focuses predominantly on different ways to integrate with OPA’s policy evaluation interface and how they compare. For more information about the management interface:

    • See the Bundle API for distributing policy and data to OPA.
    • See the for collecting status reports on bundle activation and agent health.
    • See the Decision Log API for collecting a log of policy decisions made by agents.
    • See the for checking agent deployment readiness and health.
    • See the Prometheus API endpoint to obtain insight into performance and errors.

    OPA supports different ways to evaluate policies.

    • The returns decisions as JSON over HTTP.
    • The Go API (GoDoc) returns decisions as simple Go types (, string, map[string]interface{}, etc.)
    • compiles Rego policies into WASM instructions so they can be embedded and evaluated by any WebAssembly runtime

    To integrate with OPA outside of Go, we recommend you deploy OPA as a host-level daemon or sidecar container. When your application or service needs to make policy decisions it can query OPA locally via HTTP. Running OPA locally on the same host as your application or service helps ensure policy decisions are fast and highly-available.

    Named Policy Decisions

    Use the to query OPA for named policy decisions:

    1. {
    2. "input": <the input document>
    3. }

    The <path> in the HTTP request identifies the policy decision to ask for. In OPA, every rule generates a policy decision. In the example below there are two decisions: example/authz/allow and example/authz/is_admin.

    1. package example.authz
    2. default allow = false
    3. allow {
    4. some id
    5. input.method = "GET"
    6. input.path = ["salary", id]
    7. input.subject.user = id
    8. }
    9. allow {
    10. is_admin
    11. }
    12. is_admin {
    13. }

    You can request specific decisions by querying for <package path>/<rule name>. For example to request the allow decision execute the following HTTP request:

    1. POST /v1/data/example/authz/allow
    2. Content-Type: application/json
    1. {
    2. "input": <the input document>
    3. }
    1. {
    2. "input": {
    3. "path": ["salary", "bob"],
    4. "subject": {
    5. "user": "bob"
    6. }
    7. }
    8. }

    OPA returns an HTTP 200 response code if the policy was evaluated successfully. Non-HTTP 200 response codes indicate configuration or runtime errors. The policy decision is contained in the "result" key of the response message body. For example, the above request returns the following response:

    1. 200 OK
    2. Content-Type: application/json
    1. {
    2. "result": true
    3. }

    If the requested policy decision is undefined OPA returns an HTTP 200 response without the "result" key. For example, the following request for is_admin is undefined because there is no default value for is_admin and the input does not satisfy the is_admin rule body:

    1. POST /v1/data/example/authz/is_admin
    2. Content-Type: application/json

    The response:

    1. 200 OK
    2. Content-Type: application/json
    1. {}

    For another example of how to integrate with OPA via HTTP see the HTTP API Authorization tutorial.

    Use the package to embed OPA as a library inside services written in Go. To get started import the rego package:

    1. import "github.com/open-policy-agent/opa/rego"

    The rego package exposes different options for customizing how policies are evaluated. Through the rego package you can supply policies and data, enable metrics and tracing, toggle optimizations, etc. In most cases you will:

    1. Use the rego package to construct a prepared query.
    2. Execute the prepared query to produce policy decisions.
    3. Interpret and enforce the policy decisions.

    Preparing queries in advance avoids parsing and compiling the policies on each query and improves performance considerably. Prepared queries are safe to share across multiple Go routines.

    1. module := `
    2. package example.authz
    3. default allow = false
    4. some id
    5. input.method = "GET"
    6. input.path = ["salary", id]
    7. input.subject.user = id
    8. }
    9. allow {
    10. is_admin
    11. }
    12. is_admin {
    13. input.subject.groups[_] = "admin"
    14. }
    15. `
    16. query, err := rego.New(
    17. rego.Query("x = data.example.authz.allow"),
    18. rego.Module("example.rego", module),
    19. ).PrepareForEval(ctx)
    20. if err != nil {
    21. // Handle error.
    22. }

    Using the query returned by rego.Rego#PrepareForEval call the Eval function to evaluate the policy:

    The rego.PreparedEvalQuery#Eval function returns a result set that contains the query results. If the result set is empty it indicates the query could not be satisfied. Each element in the result set contains a set of variable bindings and a set of expression values. The query from above includes a single variable x so we can lookup the value and interpret it to enforce the policy decision.

    1. if err != nil {
    2. // Handle evaluation error.
    3. } else if len(results) == 0 {
    4. // Handle undefined result.
    5. } else if result, ok := results[0].Bindings["x"].(bool); !ok {
    6. // Handle unexpected result type.
    7. } else {
    8. // Handle result/decision.
    9. }

    For more examples of embedding OPA as a library see the rego package in the Go documentation.

    Policies can be evaluated as compiled Wasm binaries.

    See for more details.

    Comparison

    A comparison of the different integration choices are summarized below.

    Integrating OPA via the REST API is the most common, at the time of writing. OPA is most often deployed either as a sidecar or less commonly as an external service. Operationally this makes it easy to upgrade OPA and to configure it to use its management services (bundles, status, decision logs, etc.). Because it is a separate process it requires monitoring and logging (though this happens automatically for any sidecar-aware environment like Kubernetes). OPA’s configuration and APIs must be secured according to the .

    Integrating via WASM is still a work-in-progress. But once it is finished, WASM policies will be embeddable in any programming language that has a WASM runtime. Evaluation will have less overhead than the REST API (because it is evaluated in the same operating-system process) and should outperform the Go API (because the policies have been compiled to a lower-level instruction set). Each programming language will need its own SDKs (also a WIP) that implement the management functionality and the evaluation interface. Typically new OPA language features will not require updating the service since neither the WASM runtime nor the SDKs will be impacted. Updating the SDKs will require re-deploying the service. Security will be analogous to the Go API integration: it is mainly the management functionality that presents security risks.