#
Signed JWT with Hosted JWKS
The following example uses NodeJS code to show how to prepare for signed JWT authentication to an API on the BC Government API Gateway.
#
1. Generate a Certificate Key Pair
npm install --save node-jose ms
echo "
const fs = require('fs');
const jose = require('node-jose');
const keyStore = jose.JWK.createKeyStore()
keyStore.generate('RSA', 2048, {alg: 'RS256', use: 'sig' })
.then(result => {
fs.writeFileSync(
'keys.json',
JSON.stringify(keyStore.toJSON(true), null, ' ')
)
const [key] = keyStore.all({ use: 'sig' })
console.log(key.toPEM(true))
console.log(key.toPEM(false))
})
" | node
#
2. Publish the Public Key
Publish the Public Key in JWKS Format at a location that is publically accessible.
echo "
const fs = require('fs')
const jose = require('node-jose');
const ks = fs.readFileSync('keys.json')
jose.JWK.asKeyStore(ks.toString()).then(keyStore =>
console.log(JSON.stringify(keyStore.toJSON(), null, 3)));
" | node
Place the JWKS JSON file somewhere that it can be reached publically.
For testing, an easy option is to use Github Pages and publish the JWKS JSON file in order to make a public URL. For example:
https://ikethecoder.github.io/temp-place/certs.json
#
3. Request Access to an API
Go to the API Services Portal and request access to an API that is configured with the Signed JWT protection. After selecting the environment, you will be prompted to provide a "JWKS URL", which will be the URL of the JWKS file that you published in step 2. After requesting access, you will be provided with some secrets.
Make a note of the Client ID
, Issuer
and Token Endpoint
.
export CID=""
export ISS=""
export TURL=""
#
4. Request a Client JWT Token
Requesting a Client JWT Token is a two-step process:
Build a Client Assertion Token that is signed with the private key you generated earlier.
Request a token from the Token Endpoint using the Client Assertion.
The following sample performs both steps:
npm install --save njwt node-fetch
echo "
const njwt = require('njwt');
const fs = require('fs')
const jose = require('node-jose');
const ks = fs.readFileSync('keys.json')
jose.JWK.asKeyStore(ks.toString()).then(keyStore => {
const [key] = keyStore.all({ use: 'sig' })
const privateKey = key.toPEM(true) // Load an RSA private key from configuration
const clientId = process.env.CID; // Or load from configuration
const now = Math.floor( new Date().getTime() / 1000 ); // seconds since epoch
const plus5Minutes = new Date( ( now + (5*60) ) * 1000); // Date object
const alg = 'RS256'; // one of RSA or ECDSA algorithms: RS256, RS384, RS512, ES256, ES384, ES512
const claims = {
aud: process.env.ISS
};
const jwt = njwt.create(claims, privateKey, alg)
.setIssuedAt(now)
.setExpiration(plus5Minutes)
.setIssuer(clientId)
.setSubject(clientId)
.compact();
const fetch = require('node-fetch');
const { URLSearchParams } = require('url');
const params = new URLSearchParams();
params.append('grant_type', 'client_credentials');
params.append('client_id', process.env.CID);
params.append('scopes', 'openid');
params.append('client_assertion_type', 'urn:ietf:params:oauth:client-assertion-type:jwt-bearer');
params.append('client_assertion', jwt);
fetch(process.env.TURL, { method: 'POST', body: params })
.then(res => res.json())
.then(json => console.log(json));
});
" | node
#
5. Call the API
Call the API using the newly generated Token returned from the Identity Provider.
curl -v https://a-protected-api.test.api.gov.bc.ca/headers \
-H "Authorization: Bearer $TOK"