Json Web Token attacks
JSON web tokens (JWTs) are a standardized format for sending cryptographically signed JSON data between systems.
The server that issues the token typically generates the signature by hashing the header and payload. In some cases, they also encrypt the resulting hash.
-
As the signature is directly derived from the rest of the token, changing a single byte of the header or payload results in a mismatched signature.
-
Without knowing the server's secret signing key, it shouldn't be possible to generate the correct signature for a given header or payload.
JSON Web Signature (JWS) -
JSON Web Encryption (JWE) - The contents of the token are encrypted.
JWT signature verification attack
1. Server not verifying the signature
If the server doesn't verify the signature properly, there's nothing to stop an attacker from making arbitrary changes to the rest of the token.
For example, consider a JWT containing the following claims:
{ "username": "carlos", "isAdmin": false }
If the server identifies the session based on this username
, modifying its value might enable an attacker to impersonate other logged-in users. Similarly, if the isAdmin
value is used for access control, this could provide a simple vector for privilege escalation.
2. Accepting arbitrary signatures
JWT libraries typically provide one method for verifying tokens and another that just decodes them. For example, the Node.js library jsonwebtoken
has verify()
and decode()
.
Occasionally, developers confuse these two methods and only pass incoming tokens to the decode()
method. This effectively means that the application doesn't verify the signature at all.
Payload can be changed with no limitation.
3. Accepting tokens with no signature
Otherwise called the "none" attack. JWTs can be signed using a range of different algorithms, but can also be left unsigned. In this case, the alg
parameter is set to none
, which indicates a so-called "unsecured JWT".
"alg" parameter can therefore be set to:
Then, the attacker can be modify the payload.
Finally, the payload part must still be terminated with a trailing dot.
4. Brute-forcing secret keys
When implementing JWT applications, developers sometimes make mistakes like forgetting to change default or placeholder secrets.
jwt secrets payloads
https://github.com/wallarm/jwt-secrets/blob/master/jwt.secrets.list
If you run the command more than once, you need to include the --show
flag to output the results.
Once you have identified the secret key, you can use it to generate a valid signature. Use JWT Editor for that (tab Keys.)
Then, send you request to Repeater. In Repeater go to JSON Web Token tab, modify the payload and click on Sign. Select your signature.
5. JWT header parameter injections
According to the JWS specification, only the alg
header parameter is mandatory. In practice, however, JWT headers (also known as JOSE headers) often contain several other parameters. The following ones are of particular interest to attackers.
-
jwk
(JSON Web Key) - Provides an embedded JSON object representing the key. -
jku
(JSON Web Key Set URL) - Provides a URL from which servers can fetch a set of keys containing the correct key. -
kid
(Key ID) - Provides an ID that servers can use to identify the correct key in cases where there are multiple keys to choose from. Depending on the format of the key, this may have a matchingkid
parameter.
5.1. Injecting self-signed JWTs via the jwk parameter
The JSON Web Signature (JWS) specification describes an optional jwk
header parameter, which servers can use to embed their public key directly within the token itself in JWK format.
How to perform the attack with Burpsuite:
5.2. Injecting self-signed JWTs via the jku parameter
Instead of embedding public keys directly using the jwk
header parameter, some servers let you use the jku
(JWK Set URL) header parameter to reference a JWK Set containing the key. When verifying the signature, the server fetches the relevant key from this URL.
A JWK Set is a JSON object containing an array of JWKs representing different keys. You can see an example of this below.
JWK Sets like this are sometimes exposed publicly via a standard endpoint, such as /.well-known/jwks.json
.
So a way to trick this validation is by creating our own set of RSA keys.
Then, have a server serving these keys.
Then, modify the payload of the JWT.
And finally, modify the head, by adding the kdi
corresponding to our crafted RSA key, and a jku
parameter pointing to our server serving the keys. With this configuration we can use 'JWT Editor' in BurpSuite to sign the new crafted JWT.