This guide will teach you how to setup and use OPAL as Python packages (python 3.7 >) with its CLI
- Install
pip install opal-client
pip install opal-server
- Run server (example):
# Run server # in secure mode -verifying client JWTs (Replace secrets with actual secrets ;-) ) export OPAL_AUTH_PRIVATE_KEY=~/opal export OPAL_AUTH_PUBLIC_KEY=~/opal.pub export OPAL_AUTH_MASTER_TOKEN="RANDOM-SECRET-STRING" # Watching a GIT repository from a webhook export OPAL_POLICY_REPO_URL=https://github.com/permitio/opal-example-policy-repo.git export OPAL_POLICY_REPO_WEBHOOK_SECRET="RANDOM-SECRET-STRING-SHARED-WITH-GITHUB" opal-server run
- Run client (example):
# Run client # authenticating with a JWT (replace 'JWT-CRYPTOGRAPHIC-CONTENT' with actual token ) export OPAL_CLIENT_TOKEN="JWT-CRYPTOGRAPHIC-CONTENT" # connect to server export OPAL_SERVER_URL=https://opal.mydomain.com:7002 # Subscribe to specific data-topics export OPAL_DATA_TOPICS=tenants/my-org,stripe_billing,tickets opal-client run
- Setup OPAL-server
- Setup OPAL-client
- Setup server and client in secure Mode
- General Points
Getting Started with OPAL is easy - we'll install our OPAL-server to manage all the OPAL-client's we deploy. We'll deploy OPAL-clients (along side policy agents).
This HOW-TO focuses on setting-up OPAL with its packages and CLI interface, this guide is better to understand the main configurations of OPAL. There's also a separate guide for setting-up OPAL from pre-built docker images.
-
Make sure your system is running Python 3.7 or higher
-
Ideally install OPAL into a clean virtual-env
-
Both opal-server and opal-client can be configured using environment-variables, .env / .ini files , and command-line options (later overrides previous).
-
Passing lists (e.g. client's
--data-topics
):- pass delimited by "," with env-vars i.e.
OPAL_DATA_TOPICS=topic1,topic2,topic3 opal-client run
- and as multi-options for cmd options i.e.
opal-client --data-topics topics1 --data-topics topics2 --data-topics topics3 run
- pass delimited by "," with env-vars i.e.
-
Top-level CLI options listed in
--help
are available under the same name as env-vars (simply convert to uppercase and replace "-" with "_", prefix with 'OPAL' ) for exampleOPAL_SERVER_PORT=1337 opal-server run
is equivalent toopal-server --server-port 1337 run
-
pip install opal-server
-
Once installed your shell will have access to the
opal-server
command. -
if the command isn't available try deactivating/activating the virtual-env
-
run
opal-server --help
to see all the options and commands -
run the
opal-server print-config
to see all the possible configuration keys and their values (as read from defaults, evn-vars, .env, .ini, and the command-line)
-
-
-
We can run the server with the run command - i.e.
opal-server run
-
Once the server is running you can check out its Open-API live docs at /docs or /redoc (These links assume you have the server running on locally the default port - localhost:7002 )
-
- The most basic way to run the server is just with a GIT repository to watch for policy-changes and get the policy from.
- Simplest of those is using a public repository, and simply polling on it (with
OPAL_POLICY_REPO_URL
andOPAL_POLICY_REPO_POLLING_INTERVAL
)#Have the opal server monitor a repo every 60 seconds OPAL_POLICY_REPO_URL=https://github.com/permitio/opal-example-policy-repo.git opal-server --policy-repo-polling-interval 60 run
-
- Better GIT watching can be achieved via configuring a webhook back to the OPAL_SERVER's webhook route. Say your server is hosted on
opal.yourdomain.com
the webhook URL will beopal.yourdomain.com/webhook
- see GitHub's guide on configuring webhooks
- use
OPAL_POLICY_REPO_WEBHOOK_SECRET
to configure a secret you can share with the webhook provider (authenticating incoming webhooks)- you can use
opal-server generate-secret
to create a cryptographically strong secret to use
- you can use
- Better GIT watching can be achieved via configuring a webhook back to the OPAL_SERVER's webhook route. Say your server is hosted on
-
- Use
POLICY_REPO_SSH_KEY
to authenticate to a private repository (see Git hosts for hot to configure the key - for example- Github SSH Key)- The passed value for key can either be a file path, or the contents of the SSH-key (with newlines replaced with '_')
- Use
OPAL_POLICY_REPO_CLONE_PATH
,OPAL_POLICY_REPO_MAIN_BRANCH
, etc. to control how the repo is cloned
- Use
In addition to policy updates (as seen in above section) the OPAL-server can also facilitate data updates, directing OPAL-clients to fetch the needed data from various sources. see how to trigger data updates guide CLI example:
For production we should set the server to work with a production server (GUNICORN) and backbone pub/sub.
-
Gunicorn
- simply use the
run
command with the--engine-type gunicorn
option.
opal-server run --engine-type gunicorn
- (run
opal-server run --help
to see more info on therun
command) - use
--server-worker-count
to control the amount of workers (default is set to cpu-count) - You can of course put another server or proxy (e.g. NGNIX, ENVOY) in front of the OPAL-SERVER, instead of or in addition to Gunicorn
- simply use the
-
Backbone Pub/Sub
- While OPAL-servers provide a lightweight websocket pub/sub channel for the clients; in order for all OPAL-servers (workers of same server, and of course servers on other nodes) to be synced (And in turn their clients to be synced) they need to connect through a shared channel - which we refer to as the backbone pub/sub or broadcast channel.
- Backbone Pub/Sub options: Kafka, Postgres LISTEN/NOTIFY, Redis
- Use the
broadcast-uri
option (orOPAL_BROADCAST_URI
env-var) to configure an OPAL-server to work with a backbone. - for example
OPAL_BROADCAST_URI=postgres://localhost/mydb opal-server run
-
Put it all together:
OPAL_BROADCAST_URI=postgres://localhost/mydb opal-server run --engine-type gunicorn
OPAL-server can run in secure mode, signing and verifying Json Web Tokens for the connecting OPAL-clients. To achieve this we need to provide the server with a private and public key pair. In addition we need to provide the server with a master-token (random secret) that the CLI (or other tools) could use to connect to ask it and generate the aforementioned signed-JWTs.
-
Generating encryption keys
- Using a utility like ssh-keygen we can easily generate the keys (on Windows try SSH-keys Windows guide)
follow the instructions to save the keys to two files.
ssh-keygen -t rsa -b 4096 -m pem
- If you created the keys with a passphrase, you can supply the passphrase to the server via the
OPAL_AUTH_PRIVATE_KEY_PASSPHRASE
option - You can provide the keys to OPAL-server via the
OPAL_AUTH_PRIVATE_KEY
andOPAL_AUTH_PUBLIC_KEY
options - in these vars You can either provide the path to the keys, or the actual strings of the key's content (with newlines replaced with "_")
- Using a utility like ssh-keygen we can easily generate the keys (on Windows try SSH-keys Windows guide)
-
Master-secret
- You can choose any secret you'd like, but to make life easier OPAL's CLI include the generate-secret command, which you can use to generate cryptographically strong secrets easily.
opal-server generate-secret
- provide the master-token via
OPAL_AUTH_MASTER_TOKEN
- You can choose any secret you'd like, but to make life easier OPAL's CLI include the generate-secret command, which you can use to generate cryptographically strong secrets easily.
-
run the server with both keys and and master-secret
# Run server # in secure mode -verifying client JWTs (Replace secrets with actual secrets ;-) ) # (Just to be clear `~` is the user's homedir) export OPAL_AUTH_PRIVATE_KEY=~/opal export OPAL_AUTH_PUBLIC_KEY=~/opal.pub export OPAL_AUTH_MASTER_TOKEN="RANDOM-SECRET-STRING" opal-server run
-
Once the server is running we can obtain a JWT identifying our client
- We can either obtain a JWT with the CLI
opal-client obtain-token $OPAL_AUTH_MASTER_TOKEN --server-url=$YOUR_SERVERS_ADDRESS
- Or we can obtain the JWT directly from the deployed OPAL server via its REST API:
This code example assumes your opal server is at https://opal.yourdomain.com and that your master token is
curl --request POST 'https://opal.yourdomain.com/token' \ --header 'Authorization: Bearer MY_MASTER_TOKEN' \ --header 'Content-Type: application/json' \ --data-raw '{ "type": "client", }'
MY_MASTER_TOKEN
. The/token
API endpoint can receive more parameters, as documented here.
- We can either obtain a JWT with the CLI
-
- Install OPAL-client
pip install opal-client
- Install a policy-agent next to the OPAL-client
- follow these instructions to install OPA
- If you want OPAL to execute OPA for you (and act as a watchdog for it) make sure it can find the
opa
program, by adding it to the $PATH. - Note: the client needs network access to this agent to be able to administer updates to it.
- Use the client's
run
command - try:
# general help commands and options opal-client --help # help for the run command opal-client run --help
- Just like the server all top-level options can be configured using environment-variables, .env / .ini files , and command-line options (later overrides previous).
- Key options:
- Use options starting with
--server
to control how the client connects to the server (mainly--server-url
to point at the server) - Use options starting with
--client-api-
options to control how the client's API service is running - Use
--data-topics
to control which topics for data updates the client would subscribe to. - Use
--policy-subscription-dirs
- Use options starting with
Unlike the server, the opal-client currently supports working only with a single worker process (so there's no need to run it with Gunicorn). This will change in future releases.
- Run the server in secure mode
- Using the master-token you assigned to the server obtain a client JWT
You can also use the REST API to obtain the token, as shown here.
opal-client obtain-token $OPAL_AUTH_MASTER_TOKEN --server-url=$YOUR_SERVERS_ADDRESS
- run the client with env-var
OPAL_CLIENT_TOKEN
or cmd-option--client-token
to pass the JWT obtained from the serverexport OPAL_CLIENT_TOKEN="JWT-TOKEN-VALUE` opal-client run