Self-hosting guide for Docker
Docker Instance
This Docker Compose project creates a zrok instance and includes a ziti controller and router. An optional Caddy container is included to provide HTTPS and reverse proxy services for the zrok API and public shares.
DNS Configuration
- A wildcard record exists for the IP address where the zrok instance will run, e.g. if your DNS zone is
share.example.com
, then your wildcard record is*.share.example.com
.
Additional DNS Configuration for Caddy TLS
The included Caddy container can automatically manage a wildcard certificate for your zrok instance. You can enable Caddy in this compose project by renaming caddy.compose.override.yml
as compose.override.yml
.
- Ensure A Caddy DNS plugin is available for your DNS provider (see github.com/caddy-dns).
- Designate A DNS zone for zrok, e.g.
example.com
orshare.example.com
and create the zone on your DNS provider's platform. - Created an API token in your DNS provider that has permission to manage zrok's DNS zone.
Create the Docker Compose Project
Create a working directory on your Docker host and save these Docker Compose project files.
Shortcut Option
-
Run this script to download the files in the current directory.
curl https://get.openziti.io/zrok-instance/fetch.bash | bash
Or, specify the Compose project directory.
curl https://get.openziti.io/zrok-instance/fetch.bash | bash -s /path/to/compose/project/dir
Manual Option
-
Get the zrok repo ZIP file.
wget https://github.com/openziti/zrok/archive/refs/heads/main.zip
-
Unzip the zrok-instance files into the project directory.
unzip -j -d . main.zip '*/docker/compose/zrok-instance/*'
Configure the Docker Compose Project Environment
Create an .env
file in the working directory.
ZROK_DNS_ZONE=share.example.com
ZROK_USER_EMAIL=me@example.com
ZROK_USER_PWD=zrokuserpw
ZITI_PWD=zitiadminpw
ZROK_ADMIN_TOKEN=zroktoken
# Caddy TLS option: rename caddy.compose.override.yml to compose.override.yml and set these vars; allow 80,443 in firewall
# plugin name for your DNS provider
CADDY_DNS_PLUGIN=cloudflare
# API token from your DNS provider
CADDY_DNS_PLUGIN_TOKEN=abcd1234
# use the staging API until you're sure everything is working to avoid hitting the rate limit
CADDY_ACME_API=https://acme-staging-v02.api.letsencrypt.org/directory
# no TLS option: publish the insecure ports to the internet and allow them in the firewall
ZROK_INSECURE_INTERFACE=0.0.0.0
ZROK_CTRL_PORT=18080
ZROK_FRONTEND_PORT=8080
ZROK_OAUTH_PORT=8081
# ziti ports must be published to the internet and allowed by firewall
ZITI_CTRL_ADVERTISED_PORT=1280
ZITI_ROUTER_PORT=3022
# configure oauth for public shares
ZROK_OAUTH_HASH_KEY=oauthhashkeysecret
ZROK_OAUTH_GITHUB_CLIENT_ID=abcd1234
ZROK_OAUTH_GITHUB_CLIENT_SECRET=abcd1234
ZROK_OAUTH_GOOGLE_CLIENT_ID=abcd1234
ZROK_OAUTH_GOOGLE_CLIENT_SECRET=abcd1234
# zrok version, e.g., 1.0.0
ZROK_IMAGE_TAG=latest
# ziti version, e.g., 1.0.0
ZITI_IMAGE_TAG=latest
Start the Docker Compose Project
-
Start the zrok instance.
The container images for zrok (including caddy) are built in this step. This provides a simple configuration to get started. You can modify the templates named like
*.envsubst
or mount a customized configuration file to mask the one that was built in.docker compose up --build --detach
Set up a User Account
This step creates a user account. You will log in to the zrok web console with the account password created in this step. The ZROK_USER_EMAIL and ZROK_USER_PWD variables are set in the .env
file. You can create more user accounts the same way by substituting a different email and password.
docker compose exec zrok-controller bash -xc 'zrok admin create account /etc/zrok-controller/config.yml ${ZROK_USER_EMAIL} ${ZROK_USER_PWD}'
Create additional users by running the command again with a different email and password.
docker compose exec zrok-controller zrok admin create account /etc/zrok-controller/config.yml <email> <password>
Enable the User Environment
You must enable each device environment with the account token obtained when the account was created. This is separate from the account password that's used to log in to the web console.
Follow the getting started guide to install the zrok CLI on some device and enable a zrok environment.
-
Configure the environment with the zrok API. Substitute the API endpoint with the one you're using, e.g.
https://zrok.${ZROK_DNS_ZONE}
.zrok config set apiEndpoint https://zrok.share.example.com
or, if not using Caddy for TLS:
zrok config set apiEndpoint http://zrok.share.example.com:18080
-
Enable an environment on this device with the account token from the previous step.
zrok enable heMqncCyxZcx
Firewall Configuration
The ziti-quickstart
and caddy
containers publish ports to all devices that use zrok shares. The zrok-controller
and zrok-frontend
containers expose ports only to the caddy
container and the Docker host's loopback interface.
Required
443/tcp
- reverse proxy handles HTTPS requests for zrok API, OAuth, and public shares (published by containercaddy
)1280/tcp
- ziti ctrl plane (published by containerziti-quickstart
)3022/tcp
- ziti data plane (published by containerziti-quickstart
)
Optional
80/tcp
- reverse proxy redirects non-HTTPS requests to443/tcp
(published by containercaddy
)
Troubleshooting
-
Check the ziti and zrok logs.
You can substitute the service container name of each to check their logs individually:
ziti-quickstart
,zrok-controller
,zrok-frontend
.docker compose logs zrok-controller
-
Check the caddy logs.
It can take a few minutes for Caddy to obtain the wildcard certificate. You can check the logs to see if there were any errors completing the DNS challenge which involves using the Caddy DNS plugin to create a TXT record in your DNS zone. This leverages the API token you provided in the
.env
file, which must have permission to create DNS records in the zrok DNS zone.docker compose logs caddy
-
zrok enable
fails certificate verification: ensure you are not using the staging API for Let's Encrypt.If you are using the staging API, you will see an error about the API certificate when you use the zrok CLI. You can switch to the production API by removing the overriding assignment of the
CADDY_ACME_API
variable. -
Validate the Caddyfile.
docker compose exec caddy caddy validate --config /etc/caddy/Caddyfile
-
Verify the correct DNS provider module was built-in to Caddy.
docker compose exec caddy caddy list-modules | grep dns.providers
-
Verify certificates.
You can check the certificates that Caddy has obtained.
docker compose exec caddy caddy list-certificates
-
Use the Caddy admin API.
You can use the Caddy admin API to check the status of the Caddy instance. The admin API is available on port
2019/tcp
inside the Docker Compose project. You can modifycompose.override.yml
to publish the port if you want to access the admin API from the Docker host or elsewhere.docker compose exec caddy curl http://localhost:2019/config/ | jq
-
My provider, e.g., Route53 doesn't give me a single API token.
As long as your DNS provider is supported by Caddy then it will work. You can modify the Caddyfile to use a different set of properties than the example. Here's how the
tls
section should look for Route53.tls {
dns {$CADDY_DNS_PLUGIN} {
access_key_id {$AWS_ACCESS_KEY_ID}
secret_access_key {$AWS_SECRET_ACCESS_KEY}
}
}.envCADDY_DNS_PLUGIN=route53
AWS_ACCESS_KEY_ID=abcd1234
AWS_SECRET_ACCESS_KEY=abcd1234