Background
I was tasked with extending an e-sports FIFA tournament service (SAFA) that had been operating primarily on Discord through a Discord bot, to the web. Since the database for users was modeled around their Discord account ID, the aim was to develop a web interface by integrating it with Discord. Thus I implemented login functionality using Discord OAuth.
Session Management Workflow
Initially, the user is logged out. When the user logs in with their discord account, an access-token and refresh-token is created, both stored as JWTs in cookies. The access token is sent with all requests via the bearer authorization header for authentication and authorization purposes, while the refresh token is only sent to a specific refresh endpoint (/refresh
). The backend verifies these tokens (JWT verification) whenever frontend (website) makes a request to it.
If verification fails for reasons other than token expiration (missing token, invalid token signature), the user is logged out. If the token has expired, the frontend silently attempts to refresh it by sending an additional request to backend at /refresh
. If the refresh is successful, new access and refresh tokens are issued, and the original request is retried. If any error occurs during this process, including token expiry, the user is logged out.
The Problem
This process works well when only one request is sent at a time. However, since the website uses client-side-rendering and is a single-page-application built with React, the web app often sends multiple ajax-requests simultaneously. This can lead to unintended session termination due to concurrent refresh attempts which logs the user out.
Why does this happen?
When the access token has expired, the first request to the backend triggers a refresh and retrieves new tokens. However, before these new tokens are updated in the frontend, other requests are sent with the old tokens. If those requests also try to refresh the tokens, Discord returns an error (403) because the refresh token has already been used. This causes the system to terminate the user’s session thereby logging them out.
Solution Options
- Extend token expiration time: This reduces the frequency of refreshes and unintended session terminations, but it introduces security risks and is not a proper solution.
- Queue requests: We queue all the requests, or we queue requests until the refresh is completed when frontend detects (before making a request) that the token is expired. This can slow down the app, especially when expiration times are short.
- Locking mechanism of Refresh Token: The backend uses locks on Refresh Tokens, ensuring that only one refresh attempt for a Refresh Token is processed at a time.
The Chosen Solution: Redis-based Locking Mechanism
I chose to implement a locking mechanism using Redis, a fast in-memory data store with built-in key expiry.
We allow the first request to refresh the token while the others wait for the result of the first request. Here’s how it works:
- The backend returns Token expiration error for the first request and the frontend then sends a refresh request to the backend at
/refresh
. - Two Redis keys are used:
refresh_token_lock:<Refresh Token>
: Indicates whether the refresh token is being processed.refresh_token_res:<Refresh Token>
: Holds the result status of the refresh attempt for the Token.
- At
/refresh
, backend first checks if arefresh_token_res:<Refresh Token>
key exists in Redis. If it does, the system returns the corresponding result status. - If the key does not exist, we check for the
refresh_token_lock:<Refresh Token>
key to see if another request has acquired the lock and is already processing the refresh token. If the lock is held, the request waits for it to be released and repeats previous step. - If no lock is present, the request acquires the lock, performs the refresh, and sets the appropriate status key in Redis. The lock is then released, and the status is returned.
This ensures that only the first request refreshes the token, while other requests wait for the result.
Conclusion
The Redis-based locking mechanism effectively handles and prevents concurrent refresh attempts, and ensures that only one request is allowed to refresh tokens at a time. This solution enhances the overall reliability and security of token management in the system.