Skip to content

Commit

Permalink
Move browser JWT logic from its own file to Crypto.ts
Browse files Browse the repository at this point in the history
  • Loading branch information
D-Pow committed Aug 17, 2024
1 parent 278e6b2 commit f4ed5d5
Show file tree
Hide file tree
Showing 3 changed files with 147 additions and 152 deletions.
148 changes: 147 additions & 1 deletion src/utils/Crypto.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,12 @@
import { byteArrayToHexString } from '@/utils/Text';
import {
byteArrayToHexString,
base64UrlEncode,
base64UrlDecode,
} from '@/utils/Text';

import type {
ValueOf,
Indexable,
} from '@/types';


Expand Down Expand Up @@ -50,3 +55,144 @@ hash.ALGOS = {
export function uuid() {
return self?.crypto?.randomUUID?.();
}


export function decodeJwt(jwt: string, { header = true, payload = false, signature = false }): string | Indexable;
export function decodeJwt(jwt: string, { header = false, payload = true, signature = false }): string | Indexable;
export function decodeJwt(jwt: string, { header = false, payload = false, signature = true }): string | Indexable;
export function decodeJwt(jwt: string, { header = true, payload = true, signature = false }): Array<string | Indexable>;
export function decodeJwt(jwt: string, { header = true, payload = false, signature = true }): Array<string | Indexable>;
export function decodeJwt(jwt: string, { header = false, payload = true, signature = true }): Array<string | Indexable>;
export function decodeJwt(jwt: string, { header = true, payload = true, signature = true }): Array<string | Indexable>;
/**
* Decodes a JWT.
*
* @param jwt - JWT to decode;
* @param [options]
* @param [options.header] - If the header should be included in return output.
* @param [options.payload] - If the payload should be included in return output.
* @param [options.signature] - If the signature should be included in return output.
*
* @see [Inspiration code]{@link https://github.com/auth0/jwt-decode/blob/main/lib/index.ts}
* @see [Related SO answer](https://stackoverflow.com/questions/38552003/how-to-decode-jwt-token-in-javascript-without-using-a-library/38552302#38552302)
* @see [JWT.io]{@link https://jwt.io}
*/
export function decodeJwt(jwt: string, {
header = false,
payload = true,
signature = false,
} = {}): string | Indexable | (string | Indexable)[] {
const jwtParsedList: string[] = jwt.split(/\./g)
.map(str => {
return base64UrlDecode(str);
});

const jwtParsedFilteredByDesiredParts = [ header, payload, signature ]
.map((entrySelect, i) => {
if (entrySelect) {
let jwtPart = jwtParsedList[i];

try {
jwtPart = JSON.parse(jwtPart);
} catch (notJsonError) {}

return jwtPart;
}
})
.filter(Boolean) as string[];

if (jwtParsedFilteredByDesiredParts.length === 1) {
return <string> jwtParsedFilteredByDesiredParts[0];
}

return <string[]> jwtParsedFilteredByDesiredParts;
}

/**
* Creates a JWT token.
*
* @see [Browser walkthrough]{@link https://stackoverflow.com/questions/47329132/how-to-get-hmac-with-crypto-web-api/47332317#47332317}
*/
export async function encodeJwt(text: string, secret: string, {
alg = 'HS256',
typ = 'JWT',
} = {}): Promise<string> {
const encodedHeader = base64UrlEncode(JSON.stringify({ alg, typ }));
const encodedPayload = base64UrlEncode(text);
let algorithm = alg
.replace(/^hs/i, 'sha')
.replace(/^\D+/gi, match => match.toLowerCase());

algorithm = algorithm.replace(/^sha/i, 'SHA-');
const algoInfo = {
name: 'HMAC',
hash: algorithm,
};

const signingKey = await globalThis.crypto.subtle.importKey(
'raw',
new TextEncoder().encode(secret),
algoInfo,
false,
[ 'sign', 'verify' ],
);
const signature = await globalThis.crypto.subtle.sign(
algoInfo.name,
signingKey,
new TextEncoder().encode(
`${encodedHeader}.${encodedPayload}`,
),
);
/*
* For some reason, `byteArrayToHexString()` doesn't work but a standard `fromCharCode()` does.
* Likewise, these don't work either:
* - byteArrayToHexString(new Uint8Array(signature));
* - byteArrayToHexString(new Uint8Array(signature));
* - base64UrlEncode(byteArrayToHexString(new Uint8Array(signature)));
* See:
* - https://stackoverflow.com/questions/9267899/arraybuffer-to-base64-encoded-string/11562550#11562550
*/
const hmacStr = base64UrlEncode(String.fromCharCode(...new Uint8Array(signature)));

const jwt = `${encodedHeader}.${encodedPayload}.${hmacStr}`;

return jwt;
}

/**
* Verifies a JWT token.
* Note: This shouldn't be done client-side as it exposes the secret.
*/
export async function verifyJwt(jwt: string, signature: Parameters<Crypto['subtle']['verify']>[2], {
alg = 'HS256',
typ = 'JWT',
secret = '',
} = {}) {
const [ encodedHeader, encodedPayload, signatureStr ] = jwt.split('.');

let algorithm = alg
.replace(/^hs/i, 'sha')
.replace(/^\D+/gi, match => match.toLowerCase());
algorithm = algorithm.replace(/^sha/i, 'SHA-');
const algoInfo = {
name: 'HMAC',
hash: algorithm,
};

const signingKey = await globalThis.crypto.subtle.importKey(
'raw',
new TextEncoder().encode(secret),
algoInfo,
false,
[ 'verify' ],
);

return await globalThis.crypto.subtle.verify(
'HMAC',
signingKey,
signature,
new TextEncoder().encode(
`${encodedHeader}.${encodedPayload}`,
),
);
}
151 changes: 0 additions & 151 deletions src/utils/Jwt.ts

This file was deleted.

File renamed without changes.

0 comments on commit f4ed5d5

Please # to comment.