The JWT Handbook 1
The JWT Handbook 1
The JWT Handbook 1
Special Thanks 4
1 Introduction 5
1.1 What is a JSON Web Token? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
1.2 What problem does it solve? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
1.3 A little bit of history . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2 Practical Applications 8
2.1 Client-side/Stateless Sessions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
2.1.1 Security Considerations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.1.1.1 Signature Stripping . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.1.1.2 Cross-Site Request Forgery (CSRF) . . . . . . . . . . . . . . . . . . 10
2.1.1.3 Cross-Site Scripting (XSS) . . . . . . . . . . . . . . . . . . . . . . . 11
2.1.2 Are Client-Side Sessions Useful? . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.1.3 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.2 Federated Identity . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
2.2.1 Access and Refresh Tokens . . . . . . . . . . . . . . . . . . . . . . . . . . . . 18
2.2.2 JWTs and OAuth2 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
2.2.3 JWTs and OpenID Connect . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20
2.2.3.1 OpenID Connect Flows and JWTs . . . . . . . . . . . . . . . . . . . 20
2.2.4 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20
2.2.4.1 Setting up Auth0 Lock for Node.js Applications . . . . . . . . . . . 21
1
7.2.2.3 RS256: RSASSA PKCS1 v1.5 using SHA-256 . . . . . . . . . . . . . 76
7.2.2.3.1 Algorithm . . . . . . . . . . . . . . . . . . . . . . . . . . . 76
7.2.2.3.1.1 EMSA-PKCS1-v1_5 primitive . . . . . . . . . . . . 78
7.2.2.3.1.2 OS2IP primitive . . . . . . . . . . . . . . . . . . . . 79
7.2.2.3.1.3 RSASP1 primitive . . . . . . . . . . . . . . . . . . . 79
7.2.2.3.1.4 RSAVP1 primitive . . . . . . . . . . . . . . . . . . . 80
7.2.2.3.1.5 I2OSP primitive . . . . . . . . . . . . . . . . . . . . 80
7.2.2.3.2 Sample code . . . . . . . . . . . . . . . . . . . . . . . . . . 81 Chapter 1
7.2.2.4 PS256: RSASSA-PSS using SHA-256 and MGF1 with SHA-256 . . 86
7.2.2.4.1 Algorithm . . . . . . . . . . . . . . . . . . . . . . . . . . . 86
7.2.2.4.1.1 MGF1: the mask generation function . . . . . . . . 87
7.2.2.4.1.2
7.2.2.4.1.3
EMSA-PSS-ENCODE primitive . . .
EMSA-PSS-VERIFY primitive . . . .
. . . . . .
. . . . . .
.
.
.
.
88
89
Introduction
7.2.2.4.2 Sample code . . . . . . . . . . . . . . . . . . . . . . . . . . 91
7.2.3 Elliptic Curve . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 94
7.2.3.1 Elliptic-Curve Arithmetic . . . . . . . . . . . . . . . . . . . . . . . . 96
7.2.3.1.1 Point Addition . . . . . . . . . . . . . . . . . . . . . . . . . 96 JSON Web Token, or JWT (“jot”) for short, is a standard for safely passing claims in space
7.2.3.1.2 Point Doubling . . . . . . . . . . . . . . . . . . . . . . . . . 97 constrained environments. It has found its way into all1 major2 web3 frameworks4 . Simplicity,
7.2.3.1.3 Scalar Multiplication . . . . . . . . . . . . . . . . . . . . . 97 compactness and usability are key features of its architecture. Although much more complex sys-
7.2.3.2 Elliptic-Curve Digital Signature Algorithm (ECDSA) . . . . . . . . 98 tems5 are still in use, JWTs have a broad range of applications. In this little handbook, we will
7.2.3.2.1 Elliptic-Curve Domain Parameters . . . . . . . . . . . . . . 100 cover the most important aspects of the architecture of JWTs, including their binary representation
7.2.3.2.2 Public and Private Keys . . . . . . . . . . . . . . . . . . . 101 and the algorithms used to construct them, while also taking a look at how they are commonly
7.2.3.2.2.1 The Discrete Logarithm Problem . . . . . . . . . . . 101 used in the industry.
7.2.3.2.3 ES256: ECDSA using P-256 and SHA-256 . . . . . . . . . 101
7.3 Future Updates . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 104
8 Annex A. Best Current Practices 105 1.1 What is a JSON Web Token?
8.1 Pitfalls and Common Attacks . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 105
8.1.1 “alg: none” Attack . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 106 A JSON Web Token looks like this (newlines inserted for readability):
8.1.2 RS256 Public-Key as HS256 Secret Attack . . . . . . . . . . . . . . . . . . . . 108
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.
8.1.3 Weak HMAC Keys . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 109
eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWV9.
8.1.4 Wrong Stacked Encryption + Signature Verification Assumptions . . . . . . . 110
TJVA95OrM7E2cBab30RMHrHDcEfxjoYZgeFONFh7HgQ
8.1.5 Invalid Elliptic-Curve Attacks . . . . . . . . . . . . . . . . . . . . . . . . . . . 111
8.1.6 Substitution Attacks . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 112 While this looks like gibberish, it is actually a very compact, printable representation of a series
8.1.6.1 Different Recipient . . . . . . . . . . . . . . . . . . . . . . . . . . . . 112 of claims, along with a signature to verify its authenticity.
8.1.6.2 Same Recipient/Cross JWT . . . . . . . . . . . . . . . . . . . . . . 114
{
8.2 Mitigations and Best Practices . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 115
"alg": "HS256",
8.2.1 Always Perform Algorithm Verification . . . . . . . . . . . . . . . . . . . . . . 115
"typ": "JWT"
8.2.2 Use Appropriate Algorithms . . . . . . . . . . . . . . . . . . . . . . . . . . . . 116
}
8.2.3 Always Perform All Validations . . . . . . . . . . . . . . . . . . . . . . . . . . 116
8.2.4 Always Validate Cryptographic Inputs . . . . . . . . . . . . . . . . . . . . . . 116
{
8.2.5 Pick Strong Keys . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 117
1 https://github.com/auth0/express-jwt
8.2.6 Validate All Possible Claims . . . . . . . . . . . . . . . . . . . . . . . . . . . . 117
2 https://github.com/nsarno/knock
8.2.7 Use The typ Claim To Separate Types Of Tokens . . . . . . . . . . . . . . . 117 3 https://github.com/tymondesigns/jwt-auth
8.2.8 Use Different Validation Rules For Each Token . . . . . . . . . . . . . . . . . 117 4 https://github.com/jpadilla/django-jwt-auth
8.3 Conclusion . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 118 5 https://en.wikipedia.org/wiki/Security_Assertion_Markup_Language
3 5
with different examples of the use of the ideas produced by the group, were available. These drafts thus are associated to his or her session. A third party (a client-side script) might be able to harvest
would later become the JWT, JWS, JWE, JWK and JWA RFCs. As of year 2016, these RFCs these items if they are stored in an unencrypted JWT, which could raise privacy concerns.
are in the standards track process and errata have not been found in them. The group is currently
inactive.
The main authors behind the specs are Mike Jones10 , Nat Sakimura11 , John Bradley12 and Joe
Hildebrand13 .
A common method for attacking a signed JWT is to simply remove the signature. Signed JWTs
are constructed from three different parts: the header, the payload, and the signature. These three
parts are encoded separately. As such, it is possible to remove the signature and then change the
header to claim the JWT is unsigned. Careless use of certain JWT validation libraries can result in
unsigned tokens being taken as valid tokens, which may allow an attacker to modify the payload at
his or her discretion. This is easily solved by making sure that the application that performs the
validation does not consider unsigned JWTs valid.
10 http://self-issued.info/
11 https://nat.sakimura.org/
12 https://www.linkedin.com/in/ve7jtb
13 https://www.linkedin.com/in/hildjj
7 9
Mitigation techniques rely on proper validation of all data passed to the backend. In particular,
any data received from clients must always be sanitized. If cookies are used, it is possible to protect
them from being accessed by JavaScript by setting the HttpOnly flag2 . The HttpOnly flag, while
useful, will not protect the cookie from CSRF attacks.
There are pros and cons to any approach, and client-side sessions are not an exception3 . Some
applications may require big sessions. Sending this state back and forth for every request (or group
of requests) can easily overcome the benefits of the reduced chattiness in the backend. A certain
balance between client-side data and database lookups in the backend is necessary. This depends
on the data model of your application. Some applications do not map well to client-side sessions.
Others may depend entirely on client-side data. The final word on this matter is your own! Run
benchmarks, study the benefits of keeping certain state client-side. Are the JWTs too big? Does
this have an impact on bandwidth? Does this added bandwidth overthrow the reduced latency in
the backend? Can small requests be aggregated into a single bigger request? Do these requests
still require big database lookups? Answering these questions will help you decide on the right
approach.
11 13
req.cart.items.push(parseInt(req.query.id));
res.cookie('cart', newCart, {
maxAge: 1000 * 60 * 60
});
res.end();
15 17
can be quite simple. We set up the Auth0 login screen using the Auth0.js library14 in all of our
sample servers. Once a user logs in to one server, he will also have access to the other servers (even
if they are not interconnected).
Although OAuth2 makes no mention of the format of its tokens, JWTs are a good match for its 14 https://github.com/auth0/auth0.js
15 https://github.com/auth0/auth0.js
requirements. Signed JWTs make good access tokens, as they can encode all the necessary data
19 21
{
"alg": "none"
}
which gets encoded to:
eyJhbGciOiJub25lIn0
Chapter 3 It is possible to add additional, user-defined claims to the header. This is generally of
limited use, unless certain user-specific metadata is required in the case of encrypted
JWTs before decryption.
The compact serialization is a Base641 URL-safe encoding of the UTF-82 bytes of the first two
JSON elements (the header and the payload) and the data, as required, for signing or encryption 3.2.1 Registered Claims
(which is not a JSON object itself). This data is Base64-URL encoded as well. These three elements
are separated by dots (“.”). • iss: from the word issuer. A case-sensitive string or URI that uniquely identifies the party
JWT uses a variant of Base64 encoding that is safe for URLs. This encoding basically that issued the JWT. Its interpretation is application specific (there is no central authority
substitutes the “+” and “/” characters for the “-” and “_” characters, respectively. managing issuers).
Padding is removed as well. This variant is known as base64url3 . Note that all references • sub: from the word subject. A case-sensitive string or URI that uniquely identifies the party
to Base64 encoding in this document refer to this variant. that this JWT carries information about. In other words, the claims contained in this JWT
The resulting sequence is a printable string like the following (newlines inserted for readability): are statements about this party. The JWT spec specifies that this claim must be unique in
the context of the issuer or, in cases where that is not possible, globally unique. Handling of
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9. this claim is application specific.
eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWV9.
TJVA95OrM7E2cBab30RMHrHDcEfxjoYZgeFONFh7HgQ • aud: from the word audience. Either a single case-sensitive string or URI or an array of such
values that uniquely identify the intended recipients of this JWT. In other words, when this
Notice the dots separating the three elements of the JWT (in order: the header, the payload, and claim is present, the party reading the data in this JWT must find itself in the aud claim or
the signature). disregard the data contained in the JWT. As in the case of the iss and sub claims, this claim
In this example the decoded header is: is application specific.
1 https://en.wikipedia.org/wiki/Base64 • exp: from the word expiration (time). A number representing a specific date and time in the
2 https://en.wikipedia.org/wiki/UTF-8 format “seconds since epoch” as defined by POSIX6 . This claims sets the exact moment from
3 https://tools.ietf.org/html/rfc4648#section-5
6 http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap04.html#tag_04_15
23 25
3.3 Unsecured JWTs 2. Decode the string using the Base64-URL algorithm. The result is the JWT header.
3. Take the string after the period from step 1.
With what we have learned so far, it is possible to construct unsecured JWTs. These are the 4. Decode the string using the Base64-URL algorithm. The result is the JWT payload.
simplest JWTs, formed by a simple (usually static) header: The resulting JSON strings may be “prettified” by adding whitespace as necessary.
{
"alg": "none"
}
3.5.1 Sample Code
and a user defined payload. For instance: function decode(jwt) {
const [headerB64, payloadB64] = jwt.split('.');
{
// These supports parsing the URL safe variant of Base64 as well.
"sub": "user123",
const headerStr = new Buffer(headerB64, 'base64').toString();
"session": "ch72gsb320000udocl363eofy",
const payloadStr = new Buffer(payloadB64, 'base64').toString();
"name": "Pretty Name",
return {
"lastpage": "/views/settings"
header: JSON.parse(headerStr),
}
payload: JSON.parse(payloadStr)
As there is no signature or encryption, this JWT is encoded as simply two elements (newlines };
inserted for readability): }
eyJhbGciOiJub25lIn0. The full example is in file coding.js of the accompanying sample code.
eyJzdWIiOiJ1c2VyMTIzIiwic2Vzc2lvbiI6ImNoNzJnc2IzMjAwMDB1ZG9jbDM2M
2VvZnkiLCJuYW1lIjoiUHJldHR5IE5hbWUiLCJsYXN0cGFnZSI6Ii92aWV3cy9zZXR0aW5ncyJ9.
An unsecured JWT like the one shown above may be fit for client-side use. For instance, if the session
ID is a hard-to-guess number, and the rest of the data is only used by the client for constructing a
view, the use of a signature is superfluous. This data can be used by a single-page web application
to construct a view with the “pretty” name for the user without hitting the backend while he gets
redirected to his last visited page. Even if a malicious user were to modify this data he or she would
gain nothing.
Note the trailing dot (.) in the compact representation. As there is no signature, it is
simply an empty string. The dot is still added, though.
In practice, however, unsecured JWTs are rare.
27 29
A signed JWT is composed of three elements: the header, the payload, and the signature (newlines
inserted for readability):
eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.
eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWV9.
TJVA95OrM7E2cBab30RMHrHDcEfxjoYZgeFONFh7HgQ
The process for decoding the first two elements (the header and the payload) is identical to the case
of unsecured JWTs. The algorithm and sample code can be found at the end of chapter 3.
{
"alg": "HS256",
"typ": "JWT"
}
{
"sub": "1234567890",
"name": "John Doe",
"admin": true
}
Signed JWTs, however, carry an additional element: the signature. This element appears after the Figure 4.1: JWS Compact Serialization
last dot (.) in the compact serialization form.
There are several types of signing algorithms available according to the JWS spec, so the way these The full details of these algorithms are shown in chapter 7.
octets are interpreted varies. The JWS specification requires a single algorithm to be supported by
all conforming implementations:
4.1.2 Practical Aspects of Signing Algorithms
• HMAC using SHA-256, called HS256 in the JWA spec.
The specification also defines a series of recommended algorithms: All signing algorithms accomplish the same thing: they provide a way to establish the authenticity
of the data contained in the JWT. How they do that varies.
• RSASSA PKCS1 v1.5 using SHA-256, called RS256 in the JWA spec.
• ECDSA using P-256 and SHA-256, called ES256 in the JWA spec. Keyed-Hash Message Authentication Code (HMAC) is an algorithm that combines a certain payload
with a secret using a cryptographic hash function3 . The result is a code that can be used to verify a
JWA is the JSON Web Algorithms spec, RFC 75182 .
message only if both the generating and verifying parties know the secret. In other words, HMACs
These algorithms will be explained in detail in chapter 7. In this chapter, we will focus on the allow messages to be verified through shared secrets.
practical aspects of their use.
The cryptographic hash function used in HS256, the most common signing algorithm for JWTs, is
The other algorithms supported by the spec, in optional capacity, are: SHA-256. SHA-256 is explained in detail in chapter 7. Cryptographic hash functions take a message
of arbitrary length and produce an output of fixed length. The same message will always produce
• HS384, HS512: SHA-384 and SHA-512 variations of the HS256 algorithm.
the same output. The cryptographic part of a hash function makes sure that it is mathematically
• RS384, RS512: SHA-384 and SHA-512 variations of the RS256 algorithm.
infeasible to recover the original message from the output of the function. In this way, cryptographic
• ES384, ES512: SHA-384 and SHA-512 variations of the ES256 algorithm.
hash functions are one-way functions that can be used to identify messages without actually sharing
• PS256, PS384, PS512: RSASSA-PSS + MGF1 with SHA256/384/512 variants.
the message. A slight variation in the message (a single byte, for instance) will produce an entirely
These are, essentially, variations of the three main required and recommended algorithms. The different output.
meaning of these acronyms will become clearer in chapter 7.
RSASSA is a variation of the RSA algorithm4 (explained in chapter 7) adapted for signatures. RSA
2 https://tools.ietf.org/html/rfc7518 is a public-key algorithm. Public-key algorithms generate split keys: one public key and one private
3 https://en.wikipedia.org/wiki/Cryptographic_hash_function
4 https://en.wikipedia.org/wiki/RSA_%28cryptosystem%29
31 33
Public-key cryptography5 allows for other usage scenarios. For instance, using a variation of the In JWS JSON Serialization form, signed JWTs are represented as printable text with JSON format
same RSA algorithm, it is possible to encrypt messages by using the public key. These messages (i.e., what you would get from calling JSON.stringify in a browser). A topmost JSON object that
can only be decrypted using the private key. This allows a many-to-one secure communications carries the following key-value pairs is required:
channel to be constructed. This variation is used for encrypted JWTs, which are discussed in
• payload: a Base64 encoded string of the actual JWT payload object.
<div id="chapter5"></div> • signatures: an array of JSON objects carrying the signatures. These objects are defined
below.
In turn, each JSON object inside the signatures array must contain the following key-value pairs:
• protected: a Base64 encoded string of the JWS header. Claims contained in this header are
protected by the signature. This header is required only if there are no unprotected headers.
If unprotected headers are present, then this header may or may not be present.
• header: a JSON object containing header claims. This header is unprotected by the signature.
If no protected header is present, then this element is mandatory. If a protected header is
present, then this element is optional.
• signature: A Base64 encoded string of the JWS signature.
In contrast to compact serialization form (where only a protected header is present), JSON serializa-
tion admits two types of headers: protected and unprotected. The protected header is validated
by the signature. The unprotected header is not validated by it. It is up to the implementation or
user to pick which claims to put in either of them. At least one of these headers must be present.
Both may be present at the same time as well.
When both protected and unprotected headers are present, the actual JOSE header is built from
the union of the elements in both headers. No duplicate claims may be present.
The following example is taken from the JWS RFC8 :
{
"payload": "eyJpc3MiOiJqb2UiLA0KICJleHAiOjEzMDA4MTkzODAsDQogIm
h0dHA6Ly9leGFtcGxlLmNvbS9pc19yb290Ijp0cnVlfQ",
"signatures": [
{
"protected": "eyJhbGciOiJSUzI1NiJ9",
"header": { "kid": "2010-12-29" },
"signature":
"cC4hiUPoj9Eetdgtv3hF80EGrhuB__dzERat0XF9g2VtQgr9PJbu3XOiZj5RZmh7AA
Figure 4.3: Many-to-one encryption uHIm4Bh-0Qc_lF5YKt_O8W2Fp5jujGbds9uJdbF9CUAr7t1dnZcAcQjbKBYNX4BAyn
RFdiuB--f_nZLgrnbyTyWzO5vRK5h6xBArLIARNPvkSjtQBMHlb1L07Qe7K0GarZRmB
_eSN9383LcOLn6_dO--xi12jzDwusC-eOkHWEsqtFZESc6BfI7noOPqvhJ1phCnvWh6
Elliptic Curve Digital Signature Algorithm (ECDSA)6 is an alternative to RSA. This algorithm IeYI2w9QOYEUipUTI8np6LbgGY9Fs98rqVt5AXLIhWkWywlVmtVrBp0igcN_IoypGlU
also generates a public and private key pair, but the mathematics behind it are different. This PQGe77Rw"
difference allows for lesser hardware requirements than RSA for similar security guarantees. },
We will study these algorithms in more detail in chapter 7. {
"protected": "eyJhbGciOiJFUzI1NiJ9",
5 https://en.wikipedia.org/wiki/Public-key_cryptography
"header": { "kid": "e9bc097a-ce51-4036-9562-d2ade882db0d" },
6 https://en.wikipedia.org/wiki/Elliptic_Curve_Digital_Signature_Algorithm
8 https://tools.ietf.org/html/rfc7515#appendix-A.6
35 37
4.2.1 HS256: HMAC + SHA-256
39 41
3. The initialization vector: some encryption algorithms require additional (usually random)
data.
4. The encrypted data (ciphertext): the actual data that is being encrypted.
5. The authentication tag: additional data produced by the algorithms that can be used to
validate the contents of the ciphertext against tampering.
As in the case of JWS and single signatures in the compact serialization, JWE supports a single
encryption key in its compact form.
Using a symmetric key to perform the actual encryption process is a common practice
when using asymmetric encryption (public/private-key encryption). Asymmetric en-
cryption algorithms are usually of high computational complexity, and thus encrypting
long sequences of data (the ciphertext) is suboptimal. One way to exploit the benefits
of both symmetric (faster) and asymmetric encryption is to generate a random key for a
symmetric encryption algorithm, then encrypt that key with the asymmetric algorithm.
This is the second element shown above, the encrypted key.
Some encryption algorithms can process any data passed to them. If the ciphertext
is modified (even without being decrypted), the algorithms may process it nonetheless.
The authentication tag can be used to prevent this, essentially acting as a signature.
This does not, however, remove the need for the nested JWTs explained above.
Having an encrypted encryption key means there are two encryption algorithms at play in the same
JWT. The following are the encryption algorithms available for key encryption:
• RSA variants: RSAES PKCS #1 v1.5 (RSAES-PKCS1-v1_5), RSAES OAEP and OAEP
+ MGF1 + SHA-256.
• AES variants: AES Key Wrap from 128 to 256-bits, AES Galois Counter Mode (GCM)
from 128 to 256-bits.
• Elliptic Curve variants: Elliptic Curve Diffie-Hellman Ephemeral Static key agreement
using concat KDF, and variants pre-wrapping the key with any of the non-GCM AES variants
above.
• PKCS #5 variants: PBES2 (password based encryption) + HMAC (SHA-256 to 512) +
non-GCM AES variants from 128 to 256-bits.
• Direct: no encryption for the encryption key (direct use of CEK).
None of these algorithms are actually required by the JWA specification. The following are the
recommended (to be implemented) algorithms by the specification:
• RSAES-PKCS1-v1_5 (marked for removal of the recommendation in the future)
• RSAES-OAEP with defaults (marked to become required in the future)
• AES-128 Key Wrap
• AES-256 Key Wrap
Figure 5.1: Signing vs encryption using public-key cryptography • Elliptic Curve Diffie-Hellman Ephemeral Static (ECDH-ES) using Concat KDF
(marked to become required in the future)
• ECDH-ES + AES-128 Key Wrap
At this point some people may ask:
43 45
• x5t: identical to JWS, except in this case the claim points to the public-key used to encrypt
the CEK.
• x5t#S256: identical to JWS, except in this case the claim points to the public-key used to
encrypt the CEK.
At the beginning of this chapter, JWE Compact Serialization was mentioned briefly. It is basically
composed of five elements encoded in printable-text form and separated by dots (.). The basic
algorithm to construct a compact serialization JWE JWT is:
1. If required by the chosen algorithm (alg claim), generate a random number of the required
size. It is essential to comply with certain cryptographic requirements for randomness when
generating this value. Refer to RFC 40863 or use a cryptographically validated random
number generator.
2. Determine the Content Encryption Key according to the key management mode4 :
Figure 5.5: Direct key agreement • For Direct Key Agreement: use the key agreement algorithm and the random number
to compute the Content Encryption Key (CEK).
• Direct Encryption: a user-defined symmetric shared key is used as the CEK (no key deriva- • For Key Agreement with Key Wrapping: use the key agreement algorithm with
tion or generation). the random number to compute the key that will be used to wrap the CEK.
• For Direct Encryption: the CEK is the symmetric key.
3. Determine the JWE Encrypted Key according to the key management mode:
• For Direct Key Agreement and Direct Encryption: the JWE Encrypted Key is
empty.
• For Key Wrapping, Key Encryption, and Key Agreement with Key Wrapping:
encrypt the CEK to the recipient. The result is the JWE Encrypted Key.
Figure 5.6: Direct key agreement
4. Compute an Initialization Vector (IV) of the size required by the chosen algorithm. If not
required, skip this step.
Although this constitutes a matter of terminology, it is important to understand the differences 5. Compress the plaintext of the content, if required (zip header claim).
between each management mode and give each one of them a convenient name. 6. Encrypt the data using the CEK, the IV, and the Additional Authenticated Data (AAD).
The result is the encrypted content (JWE Ciphertext) and Authentication Tag. The AAD is
only used for non-compact serializations.
5.1.1.2 Content Encryption Key (CEK) and JWE Encryption Key 7. Construct the compact representation as:
It is also important to understand the difference between the CEK and the JWE Encryption Key. base64(header) + '.' +
The CEK is the actual key used to encrypt the payload: an encryption algorithm takes the CEK base64(encryptedKey) + '.' + // Steps 2 and 3
and the plaintext to produce the ciphertext. In contrast, the JWE Encryption Key is either the 3 https://tools.ietf.org/html/rfc4086
encrypted form of the CEK or an empty octet sequence (as required by the chosen algorithm). An 4 5.1.1.1
47 49
{
"header": { "alg":"RSA1_5","kid":"2011-04-29" }, // Generate a few keys. You may also import keys generated from external
"encrypted_key": // sources.
"UGhIOguC7IuEvf_NPVaXsGMoLOmwvc1GyqlIKOK1nN94nHPoltGRhWhw7Zx0- const promises = [
kFm1NJn8LE9XShH59_i8J0PH5ZZyNfGy2xGdULU7sHNF6Gp2vPLgNZ__deLKx keystore.generate('oct', 128, { kid: 'example-1' }),
GHZ7PcHALUzoOegEI-8E66jX2E4zyJKx-YxzZIItRzC5hlRirb6Y5Cl_p-ko3 keystore.generate('RSA', 2048, { kid: 'example-2' }),
YvkkysZIFNPccxRU7qve1WYPxqbb2Yw8kZqa2rMWI5ng8OtvzlV7elprCbuPh keystore.generate('EC', 'P-256', { kid: 'example-3' }),
cCdZ6XDP0_F8rkXds2vE4X-ncOIM8hAYHHi29NX0mcKiRaD0-D-ljQTP-cFPg ];
wCp6X-nZZd9OHBv-B3oWh2TbqmScqXMR4gp_A"
With node-jose, key generation is a rather simple matter. All key types usable with JWE and
},
JWS are supported. In this example we create three different keys: a simple AES 128-bit key, a
{
RSA 2048-bit key, and an Elliptic Curve key using curve P-256. These keys can be used both for
"header": { "alg":"A128KW","kid":"7" },
encryption and signatures. In the case of keys that support public/private-key pairs, the generated
"encrypted_key": "6KB707dM9YTIgHtLvtgWQ8mKwboJW3of9locizkDTHzBC2IlrT1oOQ"
key is the private key. To obtain the public keys, simply call:
}
], var publicKey = key.toJSON();
"iv": "AxY8DCtDaGlsbGljb3RoZQ",
The public key will be stored in JWK format.
"ciphertext": "KDlTtXchhZTGufMYmOYGS4HffxPSUrfmqCHXaI9wOGY",
"tag": "Mz-VPPyU4RlcuYv1IwIvzw" It is also possible to import preexisting keys:
}
// where input is either a:
This JSON Serialized JWE JWT carries a single payload for two recipients. The encryption algo- // * jose.JWK.Key instance
rithm is AES-128 CBC + SHA-256, which you can get from the protected header: // * JSON Object representation of a JWK
jose.JWK.asKey(input).
{
then(function(result) {
"enc": "A128CBC-HS256"
// {result} is a jose.JWK.Key
}
// {result.keystore} is a unique jose.JWK.KeyStore
By performing the union of all claims for each recipient, the final header for each recipient is });
constructed:
// where input is either a:
First recipient:
// * String serialization of a JSON JWK/(base64-encoded)
{ // PEM/(binary-encoded) DER
"alg":"RSA1_5", // * Buffer of a JSON JWK/(base64-encoded) PEM/(binary-encoded) DER
"kid":"2011-04-29", // form is either a:
"enc":"A128CBC-HS256", // * "json" for a JSON stringified JWK
"jku":"https://server.example.com/keys.jwks" // * "pkcs8" for a DER encoded (unencrypted!) PKCS8 private key
} // * "spki" for a DER encoded SPKI public key
// * "pkix" for a DER encoded PKIX X.509 certificate
Second recipient:
// * "x509" for a DER encoded PKIX X.509 certificate
{ // * "pem" for a PEM encoded of PKCS8 / SPKI / PKIX
"alg":"A128KW", jose.JWK.asKey(input, form).
"kid":"7", then(function(result) {
"enc":"A128CBC-HS256", // {result} is a jose.JWK.Key
"jku":"https://server.example.com/keys.jwks" // {result.keystore} is a unique jose.JWK.KeyStore
} });
51 53
const key = keystore.get('example-2'); });
const options = { }, error => {
format: compact ? 'compact' : 'general', console.log(error);
contentAlg: 'A128CBC-HS256' });
};
Decryption of RSA and Elliptic Curve algorithms is analogous, using the private-key rather than
the symmetric key. If you have a keystore with the right kid claims, it is possible to simply pass
return encrypt(key, options, JSON.stringify(payload));
the keystore to the createDecrypt function and have it search for the right key. So, any of the
}
examples above can be decrypted using the exact same code:
contentAlg selects the actual encryption algorithm. Remember there are only two variants (with
jose.JWE.createDecrypt(keystore) //just pass the keystore here
different key sizes): AES CBC + HMAC SHA and AES GCM.
.decrypt(result)
.then(decrypted => {
5.2.4 ECDH-ES P-256 (Key) + AES-128 GCM (Content) decrypted.payload = JSON.parse(decrypted.payload);
console.log(`Decrypted result: ${JSON.stringify(decrypted)}`);
The API for elliptic curves is identical to that of RSA: }, error => {
console.log(error);
function encrypt(key, options, plaintext) { });
return jose.JWE.createEncrypt(options, key)
.update(plaintext)
.final();
}
function ecdhes(compact) {
const key = keystore.get('example-3');
const options = {
format: compact ? 'compact' : 'general',
contentAlg: 'A128GCM'
};
Nested JWTs require a bit of juggling to pass the signed JWT to the encryption function. Specif-
ically, the signature + encryption steps must be performed manually. Recall that these steps are
performed in that order: first signing, then encrypting. Although technically nothing prevents the
order from being reversed, signing the JWT first prevents the resulting token from being vulnerable
to signature removal attacks.
function nested(compact) {
const signingKey = keystore.get('example-3');
const encryptionKey = keystore.get('example-2');
55 57
6.1 Structure of a JSON Web Key
JSON Web Keys are simply JSON objects with a series of values that describe the parameters
required by the key. These parameters vary according to the type of key. Common parameters are:
• kty: “key type”. This claim differentiates types of keys. Supported types are EC, for elliptic
curve keys; RSA for RSA keys; and oct for symmetric keys. This claim is required.
• use: this claim specifies the intended use of the key. There are two possible uses: sig (for
Chapter 7
signature) and enc (for encryption). This claim is optional. The same key can be used for
encryption and signatures, in which case this member should not be present.
• key_ops: an array of string values that specifies detailed uses for the key. Possible values are:
sign, verify, encrypt, decrypt, wrapKey, unwrapKey, deriveKey, deriveBits. Certain JSON Web Algorithms
operations should not be used together. For instance, sign and verify are appropriate for
the same key, while sign and encrypt are not. This claim is optional and should not be used
at the same time as the use claim. In cases where both are present, their content should be
consistent.
You have probably noted that there are many references to this chapter throughout this handbook.
• alg: “algorithm”. The algorithm intended to be used with this key. It can be any of the The reason is that a big part of the magic behind JWTs lies in the algorithms employed with it.
algorithms admitted for JWE or JWS operations. This claim is optional. Structure is important, but the many interesting uses described so far are only possible due to the
algorithms in play. This chapter will cover the most important algorithms in use with JWTs today.
• kid: “key id”. A unique identifier for this key. It can be used to match a key against a kid
Understanding them in depth is not necessary in order to use JWTs effectively, and so this chapter
claim in the JWE or JWS header, or to pick a key from a set of keys according to application
is aimed at curious minds wanting to understand the last piece of the puzzle.
logic. This claim is optional. Two keys in the same key set can carry the same kid only if
they have different kty claims and are intended for the same use.
• x5u: a URL that points to a X.509 public key certificate or certificate chain in PEM encoded 7.1 General Algorithms
form. If other optional claims are present they must be consistent with the contents of the
certificate. This claim is optional.
The following algorithms have many different applications inside the JWT, JWS, and JWE specs.
• x5c: a Base64-URL encoded X.509 DER certificate or certificate chain. A certificate chain is Some algorithms, like Base64-URL, are used for compact and non-compact serialization forms.
represented as an array of such certificates. The first certificate must be the certificate referred Others, such as SHA-256, are used for signatures, encryption, and key fingerprints.
by this JWK. All other claims present in this JWK must be consistent with the values of the
first certificate. This claim is optional.
7.1.1 Base64
• x5t: a Base64-URL encoded SHA-1 thumbprint/fingerprint of the DER encoding of a X.509
certificate. The certificate this thumbprint points to must be consistent with the claims in
Base64 is a binary-to-text encoding algorithm. Its main purpose is to turn a sequence of octets into
this JWK. This claim is optional.
a sequence of printable characters, at the cost of added size. In mathematical terms, Base64 turns
• x5t#S256: identical to the x5t claim, but with the SHA-256 thumbprint of the certificate. a sequence of radix-256 numbers into a sequence of radix-64 numbers. The word base can be used
in place of radix, hence the name of the algorithm.
Other parameters, such as x, y, or d (from the example at the opening of this chapter) are specific
to the key algorithm. RSA keys, on the other hand, carry parameters such as n, e, dp, etc. The Note: Base64 is not actually used by the JWT spec. It is the Base64-URL variant
meaning of these parameters will become clear in chapter 7, where each key algorithm is explained described later in this chapter, that is used by JWT.
in detail.
To understand how Base64 can turn a series of arbitrary numbers into text, it is first necessary to be
familiar with text-encoding systems. Text-encoding systems map numbers to characters. Although
this mapping is arbitrary and in the case of Base64 can be implementation defined, the de facto
standard for Base64 encoding is RFC 46481 .
1 https://tools.ietf.org/rfc/rfc4648.txt
59 61
If the number of octets in the input data is not divisible by three, then the last portion of data to function rotr(x, n) {
encode will have less than 24 bits of data. When this is the case, zeros are added to the concatenated return (x >>> n) | (x << (32 - n));
input data to form an integral number of 6-bit groups. There are three possiblities: }
1. The full 24 bits are available as input; no special processing is performed.
function ch(x, y, z) {
2. 16 bits of input are available, three 6-bit values are formed, and the last 6-bit value gets extra
return (x & y) ^ ((~x) & z);
zeros added to the right. The resulting encoded string is padded with an extra = character to
}
make it explicit that 8 bits of input were missing.
3. 8 bits of input are available, two 6-bit values are formed, and the last 6-bit value gets extra
function maj(x, y, z) {
zeros added to the right. The resulting encoded string is padded with two extra = characters
return (x & y) ^ (x & z) ^ (y & z);
to make it explicit that 16 bits of input were missing.
}
The padding character (=) is considered optional by some implementations. Performing the steps in
the opposite order will yield the original data, regardless of the presence of the padding characters. function bsig0(x) {
return rotr(x, 2) ^ rotr(x, 13) ^ rotr(x, 22);
}
7.1.1.1 Base64-URL
function bsig1(x) {
Certain characters from the standard Base64 conversion table are not URL-safe. Base64 is a return rotr(x, 6) ^ rotr(x, 11) ^ rotr(x, 25);
convenient encoding for passing arbitrary data in text fields. Since only two characters from Base64 }
are problematic as part of the URL, a URL-safe variant is easy to implement. The + character and
the / character are replaced by the - character and the _ character. function ssig0(x) {
return rotr(x, 7) ^ rotr(x, 18) ^ (x >>> 3);
7.1.1.2 Sample Code }
The following sample implements a dumb Base64-URL encoder. The example is written with function ssig1(x) {
simplicity in mind, rather than speed. return rotr(x, 17) ^ rotr(x, 19) ^ (x >>> 10);
}
const table = [
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', These functions are defined in the specification. The rotr function performs bitwise rotation (to
'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', the right).
'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', Additionally, the algorithm requires the message to be of a predefined length (a multiple of 64);
'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', therefore padding is required. The padding algorithm works as follows:
'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x',
'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', 1. A single binary 1 is appended to the end of the original message. For example:
'8', '9', '-', '_' Original message:
]; 01011111 01010101 10101010 00111100
Extra 1 at the end:
/** 01011111 01010101 10101010 00111100 1
* @param input a Buffer, Uint8Array or Int8Array, Array
* @returns a String with the encoded values 2. An N number of zeroes is appended so that the resulting length of the message is the solution
*/ to this equation:
export function encode(input) { L = Message length in bits
let result = ""; 0 = (65 + N + L) mod 512
for(let i = 0; i < input.length; i += 3) { 3. Then the number of bits in the original message is appended as a 64-bit integer:
const remaining = input.length - i; Original message:
63 65
0x6a09e667, 7.2 Signing Algorithms
0xbb67ae85,
0x3c6ef372,
7.2.1 HMAC
0xa54ff53a,
0x510e527f,
Hash-based Message Authentication Codes (HMAC)4 make use of a cryptographic hash function
0x9b05688c,
(such as the SHA family discussed above) and a key to create an authentication code for a specific
0x1f83d9ab,
message. In other words, a HMAC-based authentication scheme takes a hash function, a message,
0x5be0cd19
and a secret-key as inputs and produces an authentication code as output. The strength of the
);
cryptographic hash function ensures that the message cannot be modified without the secret key.
Thus, HMACs serve both purposes of authentication and data integrity.
const padded = padMessage(message);
const w = new Uint32Array(64);
for(let i = 0; i < padded.length; i += 16) {
for(let t = 0; t < 16; ++t) {
w[t] = padded[i + t];
}
for(let t = 16; t < 64; ++t) {
w[t] = ssig1(w[t - 2]) + w[t - 7] + ssig0(w[t - 15]) + w[t - 16];
}
67 69
const result = hashFn( 7.2.2 RSA
append(opadSecret,
uint32ArrayToUint8Array(hashFn(append(ipadSecret, RSA is one of the most widely used cryptosystems today. It was developed in 1977 by Ron Rivest,
message), true))), Adi Shamir and Leonard Adleman, whose initials were used to name the algorithm. The key aspect
returnBytes); of RSA lies in its asymmetry: the key used to encrypt something is not the key used to decrypt
it. This scheme is known as public-key encryption (PKI), were the public key is the encryption key
return result; and the private key is the decryption key.
}
When it comes to signatures, the private key is used to sign a piece of information and the public
To verify a message against an HMAC, one simply computes the HMAC and compares the result key is used to verify that it was signed by a specific private key (without actually knowing it).
with the HMAC that came with the message. This requires knowledge of the secret-key by all
There are variations of the RSA algorithm for both signing and encryption. We will focus on the
parties: those who produce the message, and those who only want to verify it.
general algorithm first, and then we will take a look at the different variations used with JWTs.
Many cryptographic algorithms, and in particular RSA, are based on the relative difficulty of
7.2.1.1 HMAC + SHA256 (HS256) performing certain mathematical operations. RSA picks the integer factorization7 as its main
mathematical tool. Integer factorization is the mathematical problem that attempts to find numbers
Understanding Base64-URL, SHA-256, and HMAC are all that is needed to implement the HS256 that multiplied among themselves yield the original number as result. In other words, an integer’s
signing algorithm from the JWS specification. With this is mind, we can now combine all the factors are a set of pairs of integers that when multiplied yield the original integer.
sample code developed so far and construct a fully signed JWT.
integer = factor_1 x factor_2
export default function jwtEncode(header, payload, secret) {
if(typeof header !== 'object' || typeof payload !== 'object') { This problem might seem easy at first. And for small numbers, it is. Take for instance the number
throw new Error('header and payload must be objects'); 35:
} 35 = 7 x 5
if(typeof secret !== 'string') {
throw new Error("secret must be a string"); By knowing the multiplication tables of either 7 or 5 it is easy to find two numbers that yield 35
} when multiplied. A naive algorithm to find factors for an integer could be:
1. Let n be the number we want to factor.
header.alg = 'HS256'; 2. Let x be a number between 2 (inclusive) and n / 2 (inclusive).
3. Divide n by x and check whether the remainder is 0. If it is, you have found one pair of factors
const encHeader = b64(JSON.stringify(header)); (x and the quotient).
const encPayload = b64(JSON.stringify(payload)); 4. Continue performing step 3 increasing x by 1 in each iteratior until x reaches its upper bound
const jwtUnprotected = `${encHeader}.${encPayload}`; n / 2. When it does, you have found all possible factors of n.
const signature = b64(uint32ArrayToUint8Array(
hmac(sha256, 512, secret, stringToUtf8(jwtUnprotected), true))); This is essentially the brute force approach to finding factors. As you can imagine, this algorithm
is terribly inefficient.
return `${jwtUnprotected}.${signature}`; A better version of this algorithm is called trial division and sets stricter conditions
} for x. In particular, it defines x’s upper bound as sqrt(n), and, rather than increase
Note that this function performs no validation of the header or payload (other than checking to see x by 1 in each iteration, it makes x take the value of ever bigger prime numbers. It is
if they are objects). You can call this function like this: trivial to prove why these conditions make the algorithm more efficient while keeping it
correct (though out of scope for this text).
console.log(jwtEncode({}, {sub: "test@test.com"}, 'secret'));
More efficient algorithms do exist, but, as efficient as they are, even with today’s computers, cer-
Paste the generated JWT in JWT.io’s debugger5 and see how it gets decoded and validated. tain numbers are computationally infeasible to factor. The problem is compounded when certain
This function is very similar to the one used in chapter 4 as a demonstration for the signing 7 https://en.wikipedia.org/wiki/Integer_factorization
algorithm. From chapter 4:
5 https://jwt.io
71 73
It is computationally feasible to find three very large integers e, d and n that satisfy the equation 2. Apply the OS2IP primitive to the encoded message. The result is the integer message
above. The algorithm relies on the difficulty of finding d when all other numbers are known. In representative. OS2IP is the acronym for “Octet-String to Integer Primitive”.
other words, this expression can be turned into a one-way function. d can then be considered the 3. Apply the RSASP1 primitive to the integer message representative using the private key.
private-key, while n and e are the public key. The result is the integer signature representative.
4. Apply the I2OSP primitive to convert the integer signature representative to an array of
octets (the signature). I2OSP is the acronym for “Integer to Octet-String Primitive”.
7.2.2.1 Choosing e, d and n
A possible implementation in JavaScript, given the primitives mentioned above, could look like:
1. Choose two distinct prime numbers p and q. /**
• A cryptographically secure random number generator should be used to pick candidates * Produces a signature for a message using the RSA algorithm as defined
for p and q. An insecure RNG can result in an attacker finding one of these numbers. * in PKCS#1.
• As there is no way to randomly generate prime numbers, after two random numbers * @param {privateKey} RSA private key, an object with
are picked, they should pass a primality test. Deterministic primality checks can be * three members: size (size in bits), n (the modulus) and
expensive, so some implementations rely on probabilistic methods. How probable it is * d (the private exponent), both bigInts
to find a false prime needs to be considered. * (big-integer library).
• p and q should be similar in magnitude but not identical, and should differ in length by * @param {hashFn} the hash function as required by PKCS#1,
a few digits. * it should take a Uint8Array and return a Uint8Array
2. n is the result of p times q. This is the modulus from the equation above. Its number of bits * @param {hashType} A symbol identifying the type of hash function passed.
is the key length of the algorithm. * For now, only "SHA-256" is supported. See the "hashTypes"
3. Compute Euler’s totient function13 for n. Since n is a semiprime number, this is as simple as: * object for possible values.
n - (p + q - 1). We will call this value phi(n). * @param {message} A String or Uint8Array with arbitrary data to sign
4. Choose an e that meets the following criteria: * @return {Uint8Array} The signature as a Uint8Array
• 1 < e < phi(n) */
• e and phi(n) should be coprime export function sign(privateKey, hashFn, hashType, message) {
5. Pick a d that satisfies the following expression: const encodedMessage =
−1
emsaPkcs1v1_5(hashFn, hashType, privateKey.size / 8, message);
d≡e (mod φ(n)) const intMessage = os2ip(encodedMessage);
const intSignature = rsasp1(privateKey, intMessage);
Figure 7.5: RSA basic expression const signature = i2osp(intSignature, privateKey.size / 8);
return signature;
}
The public key is composed of values n and e. The private key is composed of values n and d.
Values p, q and phi(n) should be discarded or kept secret, as they can be used to help in finding d. To verify a signature:
From the equations above, it is evident that e and d are mathematically symmetric. We can rewrite 1. Apply the OS2IP primitive to the signature (an array of octets). This is the integer sig-
the equation from step 5 as: nature representative.
2. Apply the RSAVP1 primitive to the previous result. This primitive also takes the public
d . e≡1(mod φ(n)) key as input. This is the integer message representative.
3. Apply the I2OSP primitive to the previous result. This primitive takes an expected size as
Figure 7.6: Symmetry between e and d input. This size should match the length of the key’s modulus in number of octets. The result
is the encoded message.
4. Apply the EMSA-PKCS1-V1_5-ENCODE primitive to the message that is to be verified.
So now you are probably wondering how RSA is safe if we publish the values e and n; could’t we The result is another encoded message. This primitive makes use of a hash function (usually
use those values to find d? The thing about modular arithmetic is that there are multiple possible a SHA family hash function such as SHA-256). This primitive accepts an expected encoded
solutions. As long as the d we pick satisfies the equation above, any value is valid. The bigger the message length. In this case, it will be the length in octets of the RSA number n (the key
value, the harder it is to find it. So, RSA works as long as only one of the values e or d is known to length).
13 https://en.wikipedia.org/wiki/Euler%27s_totient_function#Computing_Euler.27s_totient_function
5. Compare both encoded messages (from steps 3 and 4). If they match, the signature is valid,
75 77
2. Produce the DER encoding for the following ASN.1 structure: 3. Take each xlen-i factor from each term in order. These are the octets for the result.
DigestInfo ::= SEQUENCE {
digestAlgorithm DigestAlgorithm, 7.2.2.3.2 Sample code
digest OCTET STRING
} Since RSA requires arbitrary precision arithmetic, we will be using the big-integer16 JavaScript
library.
Where digest is the result from step 1 and DigestAlgorithm is one of:
The OS2IP and I2OSP primitives are rather simple:
DigestAlgorithm ::=
AlgorithmIdentifier { {PKCS1-v1-5DigestAlgorithms} } function os2ip(bytes) {
let result = bigInt();
PKCS1-v1-5DigestAlgorithms ALGORITHM-IDENTIFIER ::= {
{ OID id-md2 PARAMETERS NULL }| bytes.forEach((b, i) => {
{ OID id-md5 PARAMETERS NULL }| // result += b * Math.pow(256, bytes.length - 1 - i);
{ OID id-sha1 PARAMETERS NULL }| result = result.add(
{ OID id-sha256 PARAMETERS NULL }| bigInt(b).multiply(
{ OID id-sha384 PARAMETERS NULL }| bigInt(256).pow(bytes.length - i - 1)
{ OID id-sha512 PARAMETERS NULL } )
} );
});
3. If the requested length of the result is less than the result of step 3 plus 11 (reqLength <
step2Length + 11), then the primitive fails to produce the result and outputs an error message return result;
(“intended encoded message length too short”). }
4. Repeat the octet 0xFF the following number of times: requested length + step2Length -
3. This array of octets is called PS. function i2osp(intRepr, expectedLength) {
5. Produce the final encoded message (EM) as (|| is the concatenation operator): if(intRepr.greaterOrEquals(bigInt(256).pow(expectedLength))) {
EM = 0x00 || 0x01 || PS || 0x00 || step2Result throw new Error('integer too large');
}
ASN.1 OIDs are usually defined in their own specifications. In other words, you will
not find the SHA-256 OID in the PKCS#1 spec. SHA-1 and SHA-2 OIDs are defined const result = new Uint8Array(expectedLength);
in RFC 356015 . let remainder = bigInt(intRepr);
for(let i = expectedLength - 1; i >= 0; --i) {
const position = bigInt(256).pow(i);
7.2.2.3.1.2 OS2IP primitive
const quotrem = remainder.divmod(position);
The OS2IP primitive takes an array of octets and outputs an integer representative. remainder = quotrem.remainder;
result[result.length - 1 - i] = quotrem.quotient.valueOf();
• Let X1 , X2 , …, Xn be the octets from first to last of the input.
}
• Compute the result as:
n−1 n−2 return result;
result=X 1⋅256 + X 2⋅256 +...+ X n−1⋅256+ X n }
Figure 7.7: OS2IP result The I2OSP primitive essentially decomposes a number into its base 25617 components.
The RSASP1 primitive is essentially a single operation, and forms the basis of the algorithm:
16 https://www.npmjs.com/package/big-integer
7.2.2.3.1.3 RSASP1 primitive
17 https://en.wikipedia.org/wiki/Positional_notation
15 https://tools.ietf.org/html/rfc3560.html
79 81
} To use this to verify JWTs, a simple wrapper is necessary:
export function jwtVerifyAndDecode(jwt, publicKey) {
const ps = new Uint8Array(expectedLength - t.length - 3);
if(!isString(jwt)) {
ps.fill(0xff);
throw new TypeError('jwt must be a string');
assert.ok(ps.length >= 8);
}
return Uint8Array.of(0x00, 0x01, ...ps, 0x00, ...t);
const split = jwt.split('.');
}
if(split.length !== 3) {
For simplicity, only SHA-256 is supported. Adding other hash functions is as simple as adding the throw new Error('Invalid JWT format');
right OIDs. }
The signPkcs1v1_5 function puts all primitives together to perform the signature:
const header = JSON.parse(unb64(split[0]));
/** if(header.alg !== 'RS256') {
* Produces a signature for a message using the RSA algorithm as defined throw new Error(`Wrong algorithm: ${header.alg}`);
* in PKCS#1. }
* @param {privateKey} RSA private key, an object with
* three members: size (size in bits), n (the modulus) and const jwtUnprotected = stringToUtf8(`${split[0]}.${split[1]}`);
* d (the private exponent), both bigInts const valid = verifyPkcs1v1_5(publicKey,
* (big-integer library). msg => sha256(msg, true),
* @param {hashFn} the hash function as required by PKCS#1, hashTypes.sha256,
* it should take a Uint8Array and return a Uint8Array jwtUnprotected,
* @param {hashType} A symbol identifying the type of hash function passed. base64.decode(split[2]));
* For now, only "SHA-256" is supported. See the "hashTypes"
* object for possible values. return {
* @param {message} A String or Uint8Array with arbitrary data to sign header: header,
* @return {Uint8Array} The signature as a Uint8Array payload: JSON.parse(unb64(split[1])),
*/ valid: valid
export function signPkcs1v1_5(privateKey, hashFn, hashType, message) { };
const encodedMessage = }
emsaPkcs1v1_5(hashFn, hashType, privateKey.size / 8, message);
For simplicity, the private and public keys must be passed as JavaScript objects with two separate
const intMessage = os2ip(encodedMessage);
numbers: the modulus (n) and the private exponent (d) for the private key, and the modulus (n)
const intSignature = rsasp1(privateKey, intMessage);
and the public exponent (e) for the public key. This is in contrast to the usual PEM Encoded18
const signature = i2osp(intSignature, privateKey.size / 8);
format. See the rs256.js file for more details.
return signature;
} It is possible to use OpenSSL to export these numbers from a PEM key.
To use this to sign JWTs, a simple wrapper is necessary: openssl rsa -text -noout -in testkey.pem
export default function jwtEncode(header, payload, privateKey) { OpenSSL can also be used to generate a RSA key from scratch:
if(typeof header !== 'object' || typeof payload !== 'object') {
openssl genrsa -out testkey.pem 2048
throw new Error('header and payload must be objects');
} You can then export the numbers from PEM format using the command shown above.
The private-key numbers embedded in the testkey.js file are from the testkey.pem file in
header.alg = 'RS256';
the samples directory accompanying this handbook. The corresponding public key is in the
pubtestkey.pem file.
const encHeader = b64(JSON.stringify(header));
const encPayload = b64(JSON.stringify(payload)); 18 https://en.wikipedia.org/wiki/Privacy-enhanced_Electronic_Mail
83 85
const intSignature = rsasp1(privateKey, intMessage); const hashed2 = sha256(m);
const signature = i2osp(intSignature, privateKey.size / 8);
6. Generate a sequence of zero-valued octets of length: intended maximum length of result minus
return signature;
salt length minus hash length minus 2.
}
const ps = new Array(intendedLength - intendedSaltLength - 2).fill(0);
To verify a signature:
7. Concatenate the result of the previous step with the octet 0x01 and the salt.
1. Apply the OS2IP primitive to the signature (an array of octets). This is the integer sig-
nature representative. const db = [...ps, 0x01, ...salt];
2. Apply the RSAVP1 primitive to the previous result. This primitive also takes the public
8. Apply the mask generation function to the result of step 5 and set the intended length of
key as input. This is the integer message representative.
this function as the length of the result from step 7 (the mask generation function accepts an
3. Apply the I2OSP primitive to the previous result. This primitive takes an expected size as
intended length parameter).
input. This size should match the length of the key’s modulus in number of octets. The result
is the encoded message. const dbMask = mgf1(hashed2, db.length);
4. Apply the EMSA-PSS-VERIFY primitive to the message that is to be verified and the
9. Compute the result of applying the XOR operation to the results of steps 7 and 8.
result of the previous step. This primitive outputs whether the signature is valid or not. This
primitive makes use of a hash function (usually a SHA family hash function such as SHA-256). const maskedDb = db.map((value, index) => {
The primitive takes a parameter that should be the number of bits in the modulus of the key return value ^ dbMask[index];
minus 1. });
export function verifyPss(publicKey, hashFn, hashType, message, signature) { 10. If the length of the result of the previous operation is not a multiple of 8, find the difference
if(signature.length !== publicKey.size / 8) { in number of bits to make it a multiple of 8 by subtracting bits, then set this number of bits
throw new Error('invalid signature length'); to 0 beginning from the left.
}
const zeroBits = 8 * intendedLength - intendedLengthBits;
const zeroBitsMask = 0xFF >>> zeroBits;
const intSignature = os2ip(signature);
maskedDb[0] &= zeroBitsMask;
const intVerification = rsavp1(publicKey, intSignature);
const verificationMessage = 11. Concatenate the result of the previous step with the result of step 5 and the octet 0xBC. This
i2osp(intVerification, Math.ceil( (publicKey.size - 1) / 8 )); is the result.
const result = [...maskedDb, ...hashed2, 0xBC];
return emsaPssVerify(hashFn,
hashType,
mgf1.bind(null, hashFn), 7.2.2.4.1.3 EMSA-PSS-VERIFY primitive
256 / 8,
publicKey.size - 1, The primitive takes three elements:
message, • The message to be verified.
verificationMessage); • The signature as an encoded integer message.
} • The intended maximum length of the encoded integer message.
This primitive can be parameterized by the following elements:
7.2.2.4.1.1 MGF1: the mask generation function
• A hash function. In the case of PS256 this SHA-256.
Mask generation functions take input of any length and produce output of variable length. Like hash • A mask generation function. In the case of PS256 this is MGF1.
functions, they are deterministic: they produce the same output for the same input. In contrast • An intended length for the salt used internally.
to hash functions, though, the length of the output is variable. The Mask Generation Function 1
These parameters are all specified by PS256, so they are not configurable and for the purposes of
(MGF1) algorithm is defined in Public Key Cryptography Standard #1 (PKCS #1)22 .
this description are considered constants.
22 https://www.ietf.org/rfc/rfc3447.txt
87 89
8. Set the leftmost 8 * expectedLength - expectedLengthBits bits from the first byte in the return false;
element computed in the last step to 0. }
const zeroBits = 8 * expectedLength - expectedLengthBits;
if(verificationMessage.length === 0) {
const zeroBitsMask = 0xFF >>> zeroBits;
return false;
db[0] &= zeroBitsMask;
}
9. Check that the leftmost expectedLength - hashLength - saltLength - 2 bytes of the el-
ement computed in the last step are 0. Also check that the first element after the group of if(verificationMessage[verificationMessage.length - 1] !== 0xBC) {
zeros is 0x01. return false;
}
const zeroCheckLength = expectedLength - (digest1.length + saltLength + 2);
if(!db.subarray(0, zeroCheckLength).every(v => v === 0) ||
const maskedLength = expectedLength - digest1.length - 1;
db[zeroCheckLength] !== 0x01) {
const masked = verificationMessage.subarray(0, maskedLength);
return false;
const digest2 = verificationMessage.subarray(maskedLength,
}
maskedLength + digest1.length);
10. Extract the salt from the last saltLength octets of the element computed in the last step
(db). const zeroBits = 8 * expectedLength - expectedLengthBits;
const zeroBitsMask = 0xFF >>> zeroBits;
const salt = db.subarray(db.length - saltLength);
if((masked[0] & (~zeroBitsMask)) !== 0) {
11. Compute a new encoded message by concatenating eigth octets of value zero, the hash com- return false;
puted in step 1, and the salt extracted in the last step. }
const m = Uint8Array.of(0, 0, 0, 0, 0, 0, 0, 0, ...digest1, ...salt);
const dbMask = mgf(maskedLength, digest2);
12. Compute the hash of the element computed in the last step. const db = masked.map((value, index) => value ^ dbMask[index]);
db[0] &= zeroBitsMask;
const expectedDigest = hashFn(m, true);
13. Compare the element computed in the last step to the second element extracted in step 4. If const zeroCheckLength = expectedLength - (digest1.length + saltLength + 2);
they match, the signature is valid, otherwise it is not. if(!db.subarray(0, zeroCheckLength).every(v => v === 0) ||
db[zeroCheckLength] !== 0x01) {
return uint8ArrayEquals(digest2, expectedDigest);
return false;
}
7.2.2.4.2 Sample code
const salt = db.subarray(db.length - saltLength);
As expected of a variant of RSASSA, most of the code required for this algorithm is already present const m = Uint8Array.of(0, 0, 0, 0, 0, 0, 0, 0, ...digest1, ...salt);
in the implementation of RS256. The only differences are the additions of the EMSA-PSS-ENCODE, const expectedDigest = hashFn(m, true);
EMSA-PSS-VERIFY, and MGF1 primitives.
export function mgf1(hashFn, expectedLength, seed) { return uint8ArrayEquals(digest2, expectedDigest);
if(expectedLength > Math.pow(2, 32)) { }
throw new Error('mask too long'); The complete example23 is available in the files ps256.js, rsassa.js, and pkcs.js. The private-
} key numbers embedded in the testkey.js file are from the testkey.pem file in the samples directory
accompanying this handbook. The corresponding public key is in the pubtestkey.pem file. For
const hashSize = hashFn(Uint8Array.of(0), true).byteLength; help in creating keys see the RS256 example.
const count = Math.ceil(expectedLength / hashSize);
23 https://github.com/auth0/jwt-handbook-samples
const result = new Uint8Array(hashSize * count);
for(let i = 0; i < count; ++i) {
const c = i2osp(bigInt(i), 4);
91 93
P+Q≡R (mod q) ( P≠Q)
( x p , y p )+( x q , y q )≡( x r , y r ) (mod q)
y q− y p
λ≡ (mod q)
x q−x p
x r ≡ λ 2−x p −x q (mod q)
y r ≡ λ ( x p −x r )− y p (mod q)
3 x 2p +a
λ≡ (mod q)
2 yp
x r ≡ λ 2−2 x p (mod q)
y r ≡ λ ( x p −x r )− y p (mod q)
25
Public domain image taken from Wikimedia. kP≡R (mod q)
In contrast to RSA, elliptic-curve algorithms are defined for specific finite fields. Of interest for EC k=k 0 +2 k 1 +22 k 2 +…+2m k m where [k 0 …k m ]∈{0,1 }
cryptography are binary and prime fields. The JWA specification uses only prime fields, so we will
focus on these. Figure 7.16: Scalar multiplication
A field, in mathematical terms, is a set of elements for which the four basic arithmetic
operations are defined: subtraction, addition, multiplication, and division. Then, the following algorithm is applied:
“Finite” means elliptic curve cryptography works on finite sets of numbers, rather than the infinite 1. Let N be the point P.
set of real numbers. 2. Let Q be the point at infinity (0, 0).
25 https://commons.wikimedia.org/wiki/File:EllipticCurveCatalog.svg 3. For i from 0 to m do:
95 97
throw new Error('header and payload must be objects'); 'fffffc', 16),
} b: bigInt('5ac635d8aa3a93e7b3ebbd55769886' +
'bc651d06b0cc53b0f63bce3c3e27d2' +
header.alg = 'ES256'; '604b', 16)
};
const encHeader = b64(JSON.stringify(header));
const encPayload = b64(JSON.stringify(payload));
const jwtUnprotected = `${encHeader}.${encPayload}`; 7.2.3.2.2 Public and Private Keys
const ecSignature = sign(privateKey, sha256, Constructing public and private keys using elliptic curves is really simple.
sha256.hashType, stringToUtf8(jwtUnprotected));
const ecR = i2osp(ecSignature.r, 32); A private key can be constructed by picking a random number between 1 and the order n of the
const ecS = i2osp(ecSignature.s, 32); base point G. In other words:
const signature = b64(Uint8Array.of(...ecR, ...ecS)); const privateKey = bigInt.randBetween(1, p256.n);
99 101
r: r,
s: s
};
}
Verification is just as simple. For a given signature (r,s):
1. Compute the digest of the message to sign using a cryptographically secure hash function.
Let this number be e. Chapter 8
2. Let c be multiplicative inverse of s modulo the order n.
3. Let u1 be e multiplied by c modulo n.
4. Let u2 be r multiplied by c modulo n.
5. Let point A be the base point G multiplied by u1 modulo q.
6. Let point B be the public key Q multiplied by u2 modulo q.
Annex A. Best Current Practices
7. Let point C be the elliptic-curve addition of points A and B (modulo q).
8. Let v be the x coordinate of point C modulo n.
9. If v is equal to r the signature is valid, otherwise it is not.
Sample implementation: Since their release, JWTs have been used in many different places. This has exposed JWTs, and
library implementations, to a number of attacks. We have mentioned some of them in the preceding
export function verify(publicKey, hashFn, hashType, message, signature) { chapters. In this section we will take a look at the current best practices for working with JWTs.
if(hashType !== hashTypes.sha256) {
throw new Error('unsupported hash type'); This section is based on the draft for JWT Best Current Practices1 from the IETF OAuth Working
} Group2 . The version of the draft used in this section is 00, dated July 19, 20173 .
// Algorithm as described in ANS X9.62-1998, 5.4 It is also important to have a basic idea of the most common representation for JWTs: the JWS
Compact Serialization format. Unserialized JWTs have two main JSON objects in them: the header
const e = bigInt(hashFn(message), 16); and the payload.
The header object contains information about the JWT itself: the type of token, the signature or
const c = signature.s.modInv(p256.n); encryption algorithm used, the key id, etc.
const u1 = e.multiply(c).fixedMod(p256.n);
const u2 = signature.r.multiply(c).fixedMod(p256.n); The payload object contains all the relevant information carried by the token. There are several
standard claims, like sub (subject) or iat (issued at), but any custom claims can be included as
const pointA = ecMultiply(p256.G, u1, p256.q); part of the payload.
const pointB = ecMultiply(publicKey.Q, u2, p256.q); 1 https://tools.ietf.org/wg/oauth/draft-ietf-oauth-jwt-bcp/
const point = ecAdd(pointA, pointB, p256.q); 2 https://tools.ietf.org/wg/oauth/
3 https://tools.ietf.org/html/draft-ietf-oauth-jwt-bcp-00
const v = point.x.fixedMod(p256.n);
103 105
Now, since this is a signed token, we are free to read it. This also means we could construct a header: {
similar token with slightly changed data in it, although in that case we would not be able to sign it "alg": "HS256",
unless we knew the signing key. Let’s say an attacker does not know the signing key, what could he "typ": "JWT"
or she do? In this type of attack, the malicious user could attempt to use a token with no signature! }
How does that work?
Then he or she escalates permissions by changing the role claim in the payload:
First, the attacker modifies the token. For example:
payload: {
header: { "sub": "joe",
alg: "none", "role": "admin"
typ: "JWT" }
},
Now, here’s the attack: the attacker proceeds to create a newly encoded JWT by using the public
payload: {
key, which is a simple string, as the HS256 shared secret! In other words, since the shared secret
sub: "joe"
for HS256 can be any string, even a string like the public key for the RS256 algorithm can be used
role: "admin"
for that.
}
Now if we go back to our hypothetical use of the jwtDecode function from before:
Encoded (newlines inserted for readability):
const publicKey = '...';
eyJhbGciOiJub25lIiwidHlwIjoiSldUIn0.eyJzdWIiOiJqb2UiLCJyb2xlIjoiYWRtaW4ifQ.
const decoded = jwtDecode(token, publicKey);
Note that this token does not include a signature ("alg": "none") and that the role claim of the
We can now clearly see the problem, the token will be considered valid! The public key will get
payload has been changed. If the attacker manages to use this token successfully, he or she may
passed to the jwtDecode function as the second argument, but rather than being used as a public
achieve an escalation of privilege attack! Why would an attack like this work? Let’s take a look at
key for the RS256 algorithm, it will be used as a shared secret for the HS256 algorithm. This is
how some hypothetical JWT library could work. Let’s say we have decoding function that looks
caused by the jwtDecode function relying on the alg claim from the header to pick the verification
like this:
algorithm for the JWT. And the attacker changed that:
function jwtDecode(token, secret) {
header: {
// (...)
"alg": "HS256", // <-- changed by the attacker from RS256
}
"typ": "JWT"
This function takes an encoded token and a secret and attempts to verify the token and then return }
the decoded data in it. If verification fails, it throws an exception. To pick the right algorithm for
Just like in the "alg": "none" case, relying on the alg claim combined with a bad or confusing
verification, the function relies on the alg claim from the header. This is where the attack succeeds.
API can result in a successful attack by a malicious user.
In the past7 , many libraries relied on this claim to pick the verification algorithm, and, as you may
have guessed, in our malicious token the alg claim is none. That means that there’s no verification Mitigations against this attack include passing an explicit algorithm to the jwtDecode function,
algorithm, and the verification step always succeeds. checking the alg claim, or using APIs that separate public-key algorithms from shared secret
algorithms.
As you can see, this is a classic example of an attack that relies on a certain ambiguity of the API
of a specific library, rather than a vulnerability in the specification itself. Even so, this is a real
attack that was possible in several different implementations in the past. For this reason, many 8.1.3 Weak HMAC Keys
libraries today report "alg": "none" tokens as invalid, even though there’s no signature in place.
There are other possible mitigations for this type of attack, the most important one being to always HMAC algorithms rely on a shared secret to produce and verify signatures. Some people assume
check the algorithm specified in the header before attempting to verify a token. Another one is to that shared secrets are similar to passwords, and in a sense, they are: they should be kept secret.
use libraries that require the verification algorithm as an input to the verification function, rather However, that is where the similarities end. For passwords, although the length is an important
than rely on the alg claim. property, the minimum required length is relatively small compared to other types of secrets. This
7 https://auth0.com/blog/critical-vulnerabilities-in-json-web-token-libraries/ is a consequence of the hashing algorithms that are used to store passwords (along with a salt) that
prevent brute force attacks in reasonable timeframes.
107 109
successfully execute an escalation of privileges attack. In particular, attackers that have ample {
time windows to perform attacks can try as many changes to encrypted data as they like, without "sub": "joe",
having the system discard the token as invalid before processing it. Other attacks may involve "role": "admin"
feeding invalid data to subsystems that expect data to be already sanitized at that point, triggering }
bugs, failures, or serving as the entry point for other types of attacks.
This token can be used against an API to perform authenticated operations. Furthermore, at least
For this reason, JSON Web Algorithms12 only defines encryption algorithms that also include data when it comes to this service, the user joe has administrator level privileges. However, there is a
integrity verification. In other words, as long as the encryption algorithm is one of the algorithms problem with this token: there is no intended recipient or even an issuer in it. What would happen
sanctioned by JWA13 , it may not be necessary for your application to stack an encrypted JWT on if a different API, different from the intended recipient this token was issued for, used the signature
top of a signed JWT. However, if you encrypt a JWT using a non-standard algorithm, you must as the only check for validity? Let’s say there’s also a user joe in the database for that service or
either make sure that data integrity is provided by that algorithm, or you will need to nest JWTs, API. The attacker could send this same token to that other service and instantly gain administrator
using a signed JWT as the innermost JWT to ensure data integrity. privileges!
Nested JWTs14 are explicitly defined and supported by the specification. Although unusual, they
may also appear in other scenarios, like sending a token issued by some party through a third party
system that also uses JWTs.
A common mistake in these scenarios is related to the validation of the nested JWT. To make
sure that data integrity is preserved, and that data is properly decoded, all layers of JWTs must
pass all validations related to the algorithms defined in their headers. In other words, even if the
outermost JWT can be decrypted and validated, it is also necessary to validate (or decrypt) all the
innermost JWTs. Failing to do so, especially in the case of an outermost encrypted JWT carrying
an innermost signed JWT, can result in the use of unverified data, with all the associated security
issues related to that.
Elliptic-curve cryptography is one of the public-key algorithm families supported by JSON Web
Algorithms15 . Elliptic-curve cryptography relies on the intractability of the elliptic-curve discrete
logarithm problem, a mathematical problem that cannot be solved in reasonable times for big
12 https://tools.ietf.org/html/rfc7518
13 https://tools.ietf.org/html/rfc7518
14 https://tools.ietf.org/html/rfc7519#section-2
Figure 8.2: Different Recipient JWT Substitution Attack
15 https://tools.ietf.org/html/rfc7518
111 113
8.2.5 Pick Strong Keys
Although this recommendation applies to any cryptographic key, it is still ignored many times. As
we have shown above, the minimum necessary length for HMAC shared secrets is often overlooked.
But even if the shared secret were long enough, it must also be fully random. A long key with a
bad level of randomness (a.k.a. “entropy”) can still be brute-forced or guessed. To ensure this is
not the case, key generating libraries should rely on cryptographic-quality pseudo-random number
generators (PRNGs) properly seeded during initialization. At best, a hardware number generator
may be used.
This recommendation applies to both shared-key algorithms and public-key algorithms. Further-
more, in the case of shared key algorithms, human-readable passwords are not considered good
enough and are vulnerable to dictionary attacks.
Some of the attacks we have discussed rely on incorrect validation assumptions. In particular, they
rely on signature validation or decryption as the only means of validation. Some attackers may get
access to correctly signed or encrypted tokens that can be used for malicious purposes, usually by
using them in unexpected contexts. The right way to prevent these attacks is to only consider a
token valid when both the signature and the content are valid. For this reason, claims such as sub
(subject), exp (expiration time), iat (issued at), aud (audience), iss (issuer), nbf (not valid before)
are of the utmost importance and should always be validated when present. If you are creating
tokens, consider adding as many claims as necessary to prevent its use in different contexts. In
general, sub, iss, aud, and exp are always useful and should be present.
Although most of the time the typ claim has a single value (JWT), it can also be used to separate
different types of application specific JWTs. This can be useful in case your system must handle
many different types of tokens. This claim can also prevent misuse of a token in a different context
Figure 8.3: Same Recipient JWT Substitution Attack by means of an additional claim check. The JWS standard explicitly allows for application-specific
values of the typ claim.
8.2 Mitigations and Best Practices 8.2.8 Use Different Validation Rules For Each Token
We’ve had a look at common attacks using JWTs, now let’s take a look at the current list of best This practice sums up many of the ones that have been enumerated before. To prevent attacks it is
practices. All of these attacks can be successfully prevented by following these recommendations. of key importance to make sure each token that is issued has very clear and specific validation rules.
This not only means using the typ claim when appropriate, or validating all possible claims such
as iss or aud, but it also implies avoiding key reuse for different tokens where possible or using
8.2.1 Always Perform Algorithm Verification different custom claims or claim formats. This way, tokens that are meant to be used in a single
place cannot be substituted by other tokens with very similar requirements.
The "alg": "none" attack and the “RS256 public-key as HS256 shared secret” attack can be In other words, rather than using the same private key for signing all kinds of tokens, consider
prevented by this mitigation. Every time a JWT is to be validated, the algorithm must be explicitly using different private keys for each subsystem of your architecture. You can also make claims
115 117