项目作者: busykoala

项目描述 :
Fastapi OPA middleware incl. auth flow.
高级语言: Python
项目地址: git://github.com/busykoala/fastapi-opa.git
创建时间: 2021-04-03T16:54:48Z
项目社区:https://github.com/busykoala/fastapi-opa

开源协议:GNU General Public License v3.0

下载


Open Policy Agent middleware for FastAPI

Table of contents

Contributors

Thanks to all the contributors below. Furthermore thanks for raising issues.



























What does fastapi-opa do

The FastAPI extension fastapi-opa allows to add login flows and integrates
Open Policy Agent to your app.

Flow Diagram

The middleware redirects the request to the identity provider. After the
authentication it validates the token. Using the token, Open Policy Agent
decides if the response has success or failure status.

Installation

  1. poetry add [--extras "graphql"] [--extras "saml"] fastapi-opa

How to get started

:bulb: checkout the wiki for an environment setup with Keycloak and Open Policy Agent:
Getting Started with FastAPI app with Authentication and Authorization

The package combines authentication and authorization with FastAPI. You can
customize the OPAMiddleware depending on your authentication flow.

Check out these examples for the most common flows:

  • OIDC: fastapi_opa.example_oidc.py
  • SAML: fastapi_opa.example_saml.py

Open Policy Agent

The middleware sends the validated and authenticated user token to Open
Policy Agent. It adds the extra attributes request_method and
request_path.

  1. {
  2. "input": {
  3. "exp": 1617466243,
  4. "iat": 1617465943,
  5. "auth_time": 1617465663,
  6. "jti": "9aacb638-70c6-4f0a-b0c8-dbc67f92e3d1",
  7. "iss": "http://localhost:8080/auth/realms/example-realm",
  8. "aud": "example-client",
  9. "sub": "ccf78dc0-e1d6-4606-99d4-9009e74e3ab4",
  10. "typ": "ID",
  11. "azp": "david",
  12. "session_state": "41640fe7-39d2-44bc-818c-a3360b36fb87",
  13. "at_hash": "2IGw-B9f5910Sll1tnfQRg",
  14. "acr": "0",
  15. "email_verified": false,
  16. "hr": "true",
  17. "preferred_username": "david",
  18. "user": "david",
  19. "subordinates": [],
  20. "request_method": "GET",
  21. "request_path": ["finance", "salary", "david"]
  22. }
  23. }

In Open Policy Agent you can create policies using user roles,
routes, request methods etc.

An example policy (from the official Open Policy Agent
docs
)
for this setup could look like this:

  1. package httpapi.authz
  2. # bob is alice's manager, and betty is charlie's.
  3. subordinates = {"alice": [], "charlie": [], "bob": ["alice"], "betty": ["charlie"]}
  4. # HTTP API request
  5. import input
  6. default allow = false
  7. # Allow users to get their own salaries.
  8. allow {
  9. some username
  10. input.request_method == "GET"
  11. input.request_path = ["finance", "salary", username]
  12. input.user == username
  13. }
  14. # Allow managers to get their subordinates' salaries.
  15. allow {
  16. some username
  17. input.request_method == "GET"
  18. input.request_path = ["finance", "salary", username]
  19. subordinates[input.user][_] == username
  20. }

Authentication flow

Use the provided interface to set up your desired authentication flow. Then
insert it into OPAMiddleware (fastapi_opa.auth.auth_interface.AuthInterface).
Consider submitting a pull request with new flows.

You can also use these ready-to-go implementations:

API key authentication

In the API key authentication a request header needs to match a given value.

  1. # Configure API keys
  2. api_key_config = APIKeyConfig(
  3. header_key="test",
  4. api_key="1234"
  5. )
  6. api_key_auth = APIKeyAuthentication(api_key_config)

In the example the header header["test"] = "1234" authenticates the request.
For Open Policy Agent, set user to APIKey and the variable client to the
client address.

OIDC authentication

The example in How to get started provides an example for
the implementation of the OIDC Authentication.

SAML authentication

For the saml implementation create your certs using
openssl req -new -x509 -days 3652 -nodes -out sp.crt -keyout sp.key and
add the keys to the sp section of your settings.json. Checkout the test
settings to get an idea (tests/test_data/saml/*.json).
Provide the path to your own settings.json and advanced_settings.json
in the SAMLAuthConfig like in the example below (don’t use the test data in
production).

  1. from fastapi_opa import OPAConfig
  2. from fastapi_opa.auth.auth_saml import SAMLAuthentication
  3. from fastapi_opa.auth.auth_saml import SAMLConfig
  4. opa_host = "http://localhost:8181"
  5. saml_config = SAMLConfig(settings_directory="./tests/test_data/saml")
  6. saml_auth = SAMLAuthentication(saml_config)
  7. opa_config = OPAConfig(authentication=saml_auth, opa_host=opa_host,
  8. accepted_methods=["id_token", "access_token"])

Upload the certificate to your identity provider. Using Keycloak as an
identity provider you need to configure encrypt assertion,
client signature required, force POST bindings on creating the client.
Also configure: Client Scopes -> role_list (saml) -> Mappers tab ->
role list -> Single Role Attribute

Custom payload enrichment

Use the interface fastapi_opa.opa.opa_config.Injectable to add
more information to the payload sent to Open Policy Agent.

Configure the injectables in the OPAConfig:

  1. class FancyInjectable(Injectable):
  2. async def extract(self, request: Request) -> List:
  3. return ["some", "custom", "stuff"]
  4. fancy_inj = FancyInjectable("fancy_key", skip_endpoints=["/health", "/api/[^/]*/test])
  5. opa_config = OPAConfig(
  6. authentication=oidc_auth, opa_host=opa_host, injectables=[fancy_inj]
  7. )

Use skip_endpoints to choose which endpoints the injectable shouldn’t affect.
To define an endpoint, specify an exact string or a regular expression.

GraphQL enrichment

For GraphQL you can use the ready to go injectable:

  1. from fastapi_opa.opa.enrichment.graphql_enrichment import GraphQLInjectable`
  2. graphql = GraphQLInjectable("gql_injectable")
  3. opa_config = OPAConfig(authentication=oidc_auth, opa_host=opa_host, injectables=[graphql])