Migrating from PyJWT¶
When migrating from PyJWT to joserfc
, there are a few key differences to be aware of.
joserfc
provides implementations for JWS (JSON Web Signature), JWE (JSON Web Encryption),
JWK (JSON Web Key), and JWT (JSON Web Token), whereas PyJWT focuses primarily on JWS and JWT.
Additionally, joserfc supports both JWT on JWS and JWT on JWE, offering more flexibility for
token handling.
jwt.encode¶
Both PyJWT and joserfc use the .encode
method to generate a JWT, but the parameter
structure differs between the two libraries.
import jwt
# jwt.encode(payload, key, algorithm)
encoded_jwt = jwt.encode({"some": "payload"}, "secret", algorithm="HS256")
from joserfc import jwt
from joserfc.jwk import OctKey
key = OctKey.import_key("secret") # use an explicit key
# jwt.encode(header, payload, key)
encoded_jwt = jwt.encode({"alg": "HS256"}, {"some": "payload"}, key)
jwt.decode¶
Similarly, both PyJWT and joserfc use the .decode
method to verify and decode a JWT,
but the parameter structure differs.
token = jwt.decode(encoded_jwt, "secret", algorithms=["HS256"])
# => {"some": "payload"}
from joserfc import jwt
from joserfc.jwk import OctKey
key = OctKey.import_key("secret")
token = jwt.decode(encoded_jwt, key)
# => token.header : {"alg": "HS256"}
# => token.claims : {"some": "payload"}
Non-plain string key¶
When using a non-plain string key (equivalent to an “oct” key) in joserfc, such as RSA, EC, or OKP keys, the library provides built-in implementations to handle these key types. This eliminates the need for manual key handling, which is required in PyJWT.
Let’s take an example using an RSA key:
import jwt
from cryptography.hazmat.primitives import serialization
from cryptography.hazmat.backends import default_backend
private_pem = b"-----BEGIN PRIVATE KEY-----\nMIGEAgEAMBAGByqGSM49AgEGBS..."
private_key = serialization.load_pem_private_key(
private_pem, password=None, backend=default_backend()
)
encoded_jwt = jwt.encode({"some": "payload"}, private_key, algorithm="RS256")
from joserfc.jwk import RSAKey
from joserfc import jwt
private_pem = b"-----BEGIN PRIVATE KEY-----\nMIGEAgEAMBAGByqGSM49AgEGBS..."
# Import the RSA key using joserfc's RSAKey
key = RSAKey.import_key(private_pem)
header = {'alg': 'RS256'}
payload = {'some': 'payload'}
encoded = jwt.encode(header, payload, key)
Claims validation¶
Both PyJWT and joserfc
provide mechanisms for claims validation, although
they differ in their approach.
In PyJWT, claims validation is performed within the .decode
method itself. When decoding
a token, you can specify options such as verify_exp
to validate the expiration time,
verify_aud
to validate the audience, and other options for additional claim validations.
Claims validation is an integral part of the decoding process.
On the other hand, joserfc
follows a different approach by separating the decoding and
claims validation steps. The .decode method in joserfc is focused solely on decoding the
token and extracting the header and payload information. Claims validation is performed
separately using claims validators.
Verify “exp”¶
import jwt
jwt.decode(encoded_jwt, options={"verify_exp": True})
from joserfc import jwt
# claims requests has built-in validators for exp, nbf, iat
claims_requests = jwt.JWTClaimsRegistry()
token = jwt.decode(encoded_jwt, key)
claims_requests.validate(token.claims)
Required claims¶
import jwt
jwt.decode(encoded_jwt, options={"require": ["exp", "iss", "sub"]})
from joserfc import jwt
claims_requests = jwt.JWTClaimsRegistry(
exp={"essential": True},
iss={"essential": True},
sub={"essential": True},
)
token = jwt.decode(encoded_jwt, key)
claims_requests.validate(token.claims)
The JWTClaimsRegistry
accepts each claim as an Individual Claims Requests
JSON object. You can learn more from Validate claims.