module Jose.Jwe
( jwkEncode
, rsaEncode
, rsaDecode
)
where
import Control.Arrow (first)
import Crypto.Cipher.Types (AuthTag(..))
import Crypto.PubKey.RSA (PrivateKey(..), PublicKey(..), generateBlinder, private_pub)
import Crypto.Random.API (CPRG)
import Data.ByteString (ByteString)
import qualified Data.ByteString as B
import qualified Data.ByteString.Char8 as BC
import Jose.Types
import qualified Jose.Internal.Base64 as B64
import Jose.Internal.Crypto
import Jose.Jwa
import Jose.Jwk
jwkEncode :: CPRG g
=> g
-> JweAlg
-> Enc
-> Jwk
-> Payload
-> (Either JwtError Jwt, g)
jwkEncode rng a e jwk payload = case jwk of
RsaPublicJwk kPub kid _ _ -> first Right $ rsaEncodeInternal rng (hdr kid) kPub bytes
RsaPrivateJwk kPr kid _ _ -> first Right $ rsaEncodeInternal rng (hdr kid) (private_pub kPr) bytes
_ -> (Left $ KeyError "Only RSA JWKs can be used for encoding", rng)
where
hdr kid = defJweHdr {jweAlg = a, jweEnc = e, jweKid = kid, jweCty = contentType}
(contentType, bytes) = case payload of
Claims c -> (Nothing, c)
Nested (Jwt b) -> (Just "JWT", b)
rsaEncode :: CPRG g
=> g
-> JweAlg
-> Enc
-> PublicKey
-> ByteString
-> (Jwt, g)
rsaEncode rng a e = rsaEncodeInternal rng (defJweHdr {jweAlg = a, jweEnc = e})
rsaEncodeInternal :: CPRG g
=> g
-> JweHeader
-> PublicKey
-> ByteString
-> (Jwt, g)
rsaEncodeInternal rng h pubKey claims = (Jwt jwe, rng'')
where
a = jweAlg h
e = jweEnc h
hdr = encodeHeader h
(cmk, iv, rng') = generateCmkAndIV rng e
(jweKey, rng'') = rsaEncrypt rng' a pubKey cmk
aad = B64.encode hdr
(ct, AuthTag sig) = encryptPayload e cmk iv aad claims
jwe = B.intercalate "." $ map B64.encode [hdr, jweKey, iv, ct, sig]
rsaDecode :: CPRG g
=> g
-> PrivateKey
-> ByteString
-> (Either JwtError Jwe, g)
rsaDecode rng pk jwt = (decode blinder, rng')
where
(blinder, rng') = generateBlinder rng (public_n $ private_pub pk)
decode b = do
checkDots
let components = BC.split '.' jwt
let aad = head components
[h, ek, iv, payload, sig] <- mapM B64.decode components
hdr <- case parseHeader h of
Right (JweH jweHdr) -> return jweHdr
Right (JwsH _) -> Left (BadHeader "Header is for a JWS")
Right UnsecuredH -> Left (BadHeader "Header is for an unsecured JWT")
Left e -> Left e
let alg = jweAlg hdr
cek <- rsaDecrypt (Just b) alg pk ek
claims <- decryptPayload (jweEnc hdr) cek iv aad sig payload
return (hdr, claims)
checkDots = case BC.count '.' jwt of
4 -> Right ()
_ -> Left $ BadDots 4