We will use the Oauth2 Proxy to make a static site private with minimal dependencies. In this case, I'm going to serve a Quarto site (my favorite static site generator), and secure it with an email whitelist (a text file with emails). There are many other authorization schemes in addition to an email whitelist, which you can read about here.
This section describes how OAuth works in the context of this example.
First, install Quarto, and run the following command from the local/
folder:
quarto render
This will create the directory _site/
with a static site in it (HTML, CSS, etc).
Create an OAuth App, but fill out the fields like this for local testing:
The oauth2/callback
path on the Callback URL is something specific to the OAuth2 proxy. It is the endpoint this proxy uses to handle the callbacks from the OAuth provider.
Make sure you store the Client ID
and Client Secret
into the enviornment variables OAUTH2_PROXY_CLIENT_ID
and OAUTH2_PROXY_CLIENT_SECRET
, respectively.
First, add your email address to emails/email_list.txt to whitelist yourself. The OAuth2 proxy uses this list to determine who is authorized to see your site. There are many other authorization schemes in addition to an email whitelist, which you can read about here.
Next, generate a cookie secret by running python -c 'import os,base64; print(base64.urlsafe_b64encode(os.urandom(32)).decode())'
and store that value in the OAUTH2_PROXY_COOKIE_SECRET
environment variable.
Next, run the following command from this directory:
docker run -v $(pwd)/_site:/app \
-v $(pwd)/emails:/site_config \
-p 4180:4180 \
quay.io/oauth2-proxy/oauth2-proxy \
--provider github \
--upstream "file:///app/#/" \
--http-address=":4180" \
--authenticated-emails-file "/site_config/email_list.txt" \
--scope user:email \
--cookie-expire 0h0m30s \
--session-cookie-minimal true \
--skip-provider-button true \
--cookie-secret $OAUTH2_PROXY_COOKIE_SECRET \
--client-id $OAUTH2_PROXY_CLIENT_ID \
--client-secret $OAUTH2_PROXY_CLIENT_SECRET \
--cookie-csrf-per-request=true \
--redirect-url="http://localhost:4180/oauth2/callback" \
--cookie-secure=false \
--cookie-csrf-expire=5m
These flags are annotated below (but you have to copy and paste the above version for it to work.)
# COPY & PASTE THE ABOVE VERSION, THIS IS JUST AN EXPLANATION
docker run -v $(pwd)/_site:/app \ # the directory with the static site
-v $(pwd)/emails:/site_config \ # the dirctory with the email list
-p 4180:4180 # bind the ports
quay.io/oauth2-proxy/oauth2-proxy \ # the official docker image for Oauth2 proxy
--provider github \ # use GitHub as the Oauth provider
--upstream "file:///app/#/" \ # The location of the static site files
--http-address=":4180" \ # Bind the 4180 port on all interfaces which is necessary for Docker (we aren't using https for local testing).
--authenticated-emails-file "/site_config/email_list.txt" \ # This is the email whitelist
--scope user:email \ # This tells the Oauth provider, GitHub, to share your email with the Oauth proxy
--cookie-expire 0h0m30s \ # Optional: This helps the cookie expire more quickly which could be helpful for security
--session-cookie-minimal true \ # Optional: don't store uncessary info in cookie since we aren't using features that require it
--skip-provider-button true \ # Don't need a seperate "login with GitHub" screen
--cookie-secret $OAUTH2_PROXY_COOKIE_SECRET \ # This is the secret you generate, see https://oauth2-proxy.github.io/oauth2-proxy/docs/configuration/overview
--client-id $OAUTH2_PROXY_CLIENT_ID \ # This is the ID of your Oauth App from GitHub
--client-secret $OAUTH2_PROXY_CLIENT_SECRET \ # This is the secret of your Oauth App from GitHub
--cookie-csrf-per-request=true \ # allows for parallel requests
# THE BELOW FLAGS ARE ONLY FOR LOCAL TESTING \
--redirect-url="http://localhost:4180/oauth2/callback" \ # this is necessary for local testing only
--cookie-secure=false \ # this is necessary for local testing only
--cookie-csrf-expire=5m # this is necessary for local testing only
Note how the OAuth2 Proxy doubles as a web server also! That is what --upstream
flag enables.1
Preview your site at http://localhost:4180.
There is a file named emails/email_list.txt that contains a list of the email identities that are allowed to view your site. Try misspelling your email on purpose and see what happens when you do a hard refresh after a few seconds. Try changing it back.
If you are curious how all this works, read this.
Now that you have a minimal idea of how this works locally, you can proceed to host your static site on a VM or other hosted solution.
👉 See Lesson 2: Serving Your Site. 👈
In that lesson, we will show you how to:
- Use a hosting provider to deploy your secure static site, for free.
- Enable
https
and set up a custom domain.
No! You can put applications behind the proxy, like a dashboard. Instead of passing file://...
to the --upstream
flag, you pass a URL, like http://your.internal.app
. You can see an example of this on Kuberenetes here and you can read more about this in the docs on configuring the upstream.
With regards to security, OAuth is often used for authentication2, or identifying who you are. You may have seen sites that have you sign in with GitHub, Google, etc. These OAuth providers pass your identity (and other information if you consent) to the site you are trying to access. In our case, we just want to verify a user's email address so we can determine if they are allowed to see our site.
There is another step in the security flow referred to as authorization which only shows content the verified user is allowed to see (which is typically implemented in your website's code). Thankfully, Oauth2 Proxy offers many ways of authorization that's built-in. This is nice as it doesn't force you to change your website's code (like adding login forms, conditional logic, etc).
A common reason for using OAuth for authentication is to save users from creating an additional username/password for a site.
Footnotes
-
For performance purposes, you may want to put the proxy behind another webserver like Caddy or Nginx, but it's not required in this example. ↩
-
OAuth may be used for more than authentication. OAuth allows a user to grant third parties access to query data or interact with APIs relevant to the OAuth provider. For example, when a user signs into your website with GitHub, OAuth gives your app (which is the OAuth2-proxy) a token with a scope (permission level) that the user consents to. In this example, this is what the consent screen looks like, which indicates that the app wishes to get your email:
You can set other scopes for the OAuth application. In our case, we only want to verify the user's email address so we set the scope touser:email
with the argument--scope user:email
. ↩