Token revocation API
One more protective instrument in Centrifugo PRO is API to manage token revocations.
At the moment Centrifugo provides two ways to revoke tokens:
- Revoke token by ID: based on jti claim in the case of JWT.
- Revoke all user's tokens issued before certain time: based on iat in the case of JWT.
When token is revoked client with such token will be disconnected from Centrifugo shortly. And attempt to connect with a revoked token won't succeed.
How it works
By default, information about token revocations shared throughout Centrifugo cluster and kept in a process memory. So token revocation information will be lost upon Centrifugo restart.
But it's possible to enable revocation information persistence by configuring a persistence storage – in this case token revocation information will survive Centrifugo restarts.
Centrifugo also automatically expires entries in the storage to keep working set reasonably small. Keeping pool of revoked tokens small allows avoiding expensive database lookups on every check – information is loaded periodically from the database and all checks performed over in-memory data structure – thus token revocation checks are cheap and have a small impact on the overall system performance.
Configure
Token revocation features (both revocation by token ID and user token invalidation by issue time) are enabled by default in Centrifugo PRO (as soon as your JWTs has jti
and iat
claims you will be able to use revocation APIs). By default revocation information kept in a process memory.
There are two types of persistent engines supported at the moment:
redis
database
Redis persistence engine
Revocation data can be kept in Redis. To enable this configuration should be:
{
"user_tokens_invalidate": {
"storage_type": "redis",
"redis": {
"address": "localhost:6379"
}
},
"token_revoke": {
"storage_type": "redis",
"redis": {
"address": "localhost:6379"
}
}
}
Unlike many other Redis features in Centrifugo consistent sharding is not supported for revocation data. The reason is that we don't want to loose revocation information when additional Redis node added. So only one Redis shard can be provided for token_revoke
and user_tokens_invalidate
features. This should be fine given that working set of revoked entities should be reasonably small and old entries expire. If you try to set several Redis shards here Centrifugo will exit with an error on start.
One more thing you may notice is that Redis configuration here does not have use_redis_from_engine
option. The reason is that since Redis is not shardable here reusing Redis configuration here could cause problems at the moment of main Redis scaling – which we want to avoid thus require explicit configuration here.
Database persistence engine
Revocation data can be kept in the relational database. Only PostgreSQL is supported.
To enable this configuration should be like:
{
"database": {
"enabled": true,
"postgresql": {
"dsn": "postgresql://postgres:[email protected]:5432/postgres"
}
},
"user_tokens_invalidate": {
"storage_type": "database"
},
"token_revoke": {
"storage_type": "database"
}
}
Revoke token API
revoke_token
Allows revoking individual tokens. For example, this may be useful when token leakage has been detected and you want to revoke access for a particular tokens. BTW Centrifugo PRO provides user_connections
API which has an information about tokens for active users connections (if set in JWT).
This API assumes that JWTs you are using contain "jti"
claim which is a unique token ID (according to RFC).
Example:
curl --header "Content-Type: application/json" \
--header "X-API-Key: <API_KEY>" \
--request POST \
--data '{"uid": "xxx-xxx-xxx", "expire_at": 1635845122}' \
http://localhost:8000/api/revoke_token
revoke_token params
Parameter name | Parameter type | Required | Description |
---|---|---|---|
uid | string | yes | Token unique ID (JTI claim in case of JWT) |
expire_at | int | no | Unix time in the future when revocation information should expire (Unix seconds). While optional we recommend to use a reasonably small expiration time (matching the expiration time of your JWTs) to keep working set of revocations small (since Centrifugo nodes periodically load all entries from the database table to construct in-memory cache). |
revoke_token result
Empty object at the moment.
Invalidate user tokens API
invalidate_user_tokens
Allows revoking all tokens for a user which were issued before a certain time. For example, this may be useful after user changed a password in an application.
This API assumes that JWTs you are using contain "iat"
claim which is a time token was issued at (according to RFC).
Example:
curl --header "Content-Type: application/json" \
--header "X-API-Key: <API_KEY>" \
--request POST \
--data '{"user": "test", "issued_before": 1635845022, "expire_at": 1635845122}' \
http://localhost:8000/api/invalidate_user_tokens
InvalidateUserTokensRequest
Parameter name | Parameter type | Required | Description |
---|---|---|---|
user | string | yes | User ID to invalidate tokens for |
issued_before | int | no | All tokens issued at before this Unix time will be considered revoked (in case of JWT this requires iat to be properly set in JWT), if not provided server uses current time |
expire_at | int | no | Unix time in the future when revocation information should expire (Unix seconds). While optional we recommend to use a reasonably small expiration time (matching the expiration time of your JWTs) to keep working set of revocations small (since Centrifugo nodes periodically load all entries from the database table to construct in-memory cache). |
InvalidateUserTokensRequestResult
Empty object.