Document updated on Apr 1, 2020
The API Gateway authorizes users that provide valid tokens according to your criteria, but at some point, you might want to change your mind and decide to revoke JWT tokens that are still valid.
When are you going to need this? Examples of situations where you might need to revoke perfectly legit tokens: As a user, I want to log me out of all my devices. As an administrator, I want to kick out someone from the platform. As a software releaser, my new backend version requires new fields in the tokens. I want all my sessions renegotiated again, or all users of a specific app (Android, iOS, Web app, etc.) have to be invalidated.
KrakenD integrates the bloom filter component that allows you to store in an optimized way tokens to revoke on the subsequent requests.
When you enable the bloom filter, it inspects the payload of incoming JWT tokens to check if any of the configured fields in token_keys
contains a blocked value. And if a blocked is found, access is not permitted.
The bloom filter component brings the following functionalities:
The communication with the bloom filter is RPC-based. The component exposes a listening port of your choice to receive updates of the bloom filter (single or batch), but a client is needed to communicate with the component.
The bloom filter library includes a client, so you can send the updates. In addition, the KrakenD Playground project consists of a sample web page with a form and an RPC client that sends commands to the bloom filter and updates it.
The Bloom filter is ideal for supporting a massive rejection of tokens with very little memory consumption. For instance, 100 million tokens of any size consume around 0.5GB RAM (with a rate of false positives of 1 in 999,925,224 tokens), and lookups complete in constant time (k number of hashes). These numbers are impossible to get with a key-value or a relational database.
The tokens are in-memory and directly in the rejecter interface, so the system quickly resolves the match.
The bloom filter lives at the extra_config
in the root level of the configuration, using the namespace auth/revoker
:
{
"version": "2",
"name": "My lovely gateway",
"extra_config":{
"auth/revoker": {
"N": 10000000,
"P": 0.0000001,
"hash_name": "optimal",
"TTL": 1500,
"port": 1234,
"token_keys": ["jti"]
}
}
}
All the configuration fields are mandatory, and are explained below:
N
: The maximum number of elements that you want to keep in memory.P
: The probability of returning a false positive.hash_name
: Either optimal
(recommended) or default
.TTL
: The lifespan of the JWT you are generating in seconds. The value must match the expiration you are setting in the backend.port
: The port number exposed (has to be free) for the RPC service to communicate with the bloomfilter.token_keys
: The list with all the fields in your JWT payload that need watching. These fields establish the criteria to revoke accesses in the future.The values N
and P
determine the size of the resulting bloom filter to fulfill your expectations. You can use this bloom filter calculator to play with the numbers.
Our sample JWT payload has the following characteristics:
{
"aud": "https://www.krakend.io",
"iss": "https://api.krakend.io",
"sub": "[email protected]",
"jti": "mnb23vcsrt756yuiomnbvcx98ertyuiop",
"roles": ["user", "premium"],
"did": "Android 8.0.0",
"exp": 1735689600
}
The following list shows the possible functionalities with an example"token_keys": ["jti","sub","did","aud"]
:
jti
to revoke a single user session and devicesub
to revoke all sessions of the same subject.did
to revoke all sessions using the same device ID (e.g., a new release in the Play Store)aud
to revoke all our users of this audience or application.Options are endless; these are some random examples, but it’s up to you to decide which are the JWT elements you want to watch and apply revocations. If, for instance, you only want to revoke access to a particular user or session, you only need to look at the jti
(the unique identifier of a user) and sub
.
All KrakenD nodes are stateless and act individually; they don’t synchronize. Every node needs to receive the RPC notification about any tokens that need insertion in every local bloom filter.
The bloom filter gets updated while the service is running, but the level of synchronization between the nodes depends on your push strategy to the different cluster members. KrakenD uses conflict-free replicated data types (CRDT), so you can replicate the data across multiple computers in a network without coordination between the replicas, and where it is always mathematically possible to resolve inconsistencies that might result.
The resulting system is eventually consistent.
The bloom filter management is brought to you by the component, and for the administration part, the client offers the necessary tools to adapt the gateway to your scenario. The implementation very much depends on what you want to achieve.
If you want to learn bloomfilters by example, have a look at the following resources:
The documentation is only a piece of the help you can get! Whether you are looking for Open Source or Enterprise support, see more support channels that can help you.