generated from drivly/worker.templates.do
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathworker.js
125 lines (114 loc) · 4.92 KB
/
worker.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
import { jwtVerify, SignJWT } from 'jose'
import { nanoid } from 'nanoid'
import qs from 'qs'
export const api = {
icon: '🚀',
name: 'jwt.do',
description: 'JWT Token Generation & Verification API',
url: 'https://jwt.do/api',
type: 'https://apis.do/security',
endpoints: {
generate: 'https://jwt.do/generate',
verify: 'https://jwt.do/verify',
},
site: 'https://jwt.do',
login: 'https://jwt.do/login',
signup: 'https://jwt.do/signup',
subscribe: 'https://jwt.do/subscribe',
repo: 'https://github.com/drivly/jwt.do',
}
export const gettingStarted = [`If you don't already have a JSON Viewer Browser Extension, get that first:`, `https://extensions.do`]
export const examples = {
generate: 'https://jwt.do/generate?profile[id]=1234&secret=secret&issuer=jwt.do&scope=user:read&expirationTTL=2h',
verify: 'https://jwt.do/verify?token=:token&secret=secret&issuer=jwt.do',
}
export default {
fetch: async (req, env) => {
let user = { authenticated: false }
try {
const url = new URL(req.url)
let query = (url.search && qs.parse(url.search.substring(1))) || {}
const apikey = extractKey(req, query)
let claims = (apikey && (await extractKeyClaims(req, env, apikey))) || (await extractCookieClaims(req, env)) || {}
query = { ...query, ...claims }
if (query.profile) {
if (env.ADMIN_IDS?.split(',')?.includes(query.profile.id)) {
query.profile.role = 'admin'
} else if (query.profile.role === 'admin') {
delete query.profile.role
}
user = { authenticated: true, ...query.profile }
}
if (url.pathname === '/generate') return json({ api, token: await generate(query), user })
else if (url.pathname === '/verify') return json({ api, jwt: await verify(query), user })
else return json({ api, gettingStarted, examples, user })
} catch (error) {
return json({ api, error, user }, 400)
}
},
}
const json = (obj, status) => new Response(JSON.stringify(obj, null, 2), { headers: { 'content-type': 'application/json; charset=utf-8' }, status })
function extractKey(req, query) {
const apikey = query.apikey
if (apikey) {
delete query.apikey
return apikey
}
const auth = req.headers.get('authorization')?.split(' ')
return req.headers.get('x-api-key') || auth?.[1] || auth?.[0]
}
async function extractKeyClaims(req, env, apikey) {
const domain = extractDomain(new URL(req.url))
if (domain === 'apikeys.do') return
const { profile } = await env.APIKEYS.fetch(new Request('/api?apikey=' + apikey, req.url)).then((res) => res.json())
return profile && { secret: env.JWT_SECRET + domain, profile, issuer: domain }
}
async function extractCookieClaims(req, env) {
const url = new URL(req.url)
const domain = extractDomain(url)
const secret = env.JWT_SECRET + domain
const cookie = req.headers.get('cookie')
const cookies = cookie && Object.fromEntries(cookie.split(';').map((c) => c.trim().split('=')))
const token = cookies?.['__Secure-worker.auth.providers-token']
if (!token) return
try {
const jwt = await verify({ token, secret, issuer: domain })
const { profile } = jwt.payload
return { secret, profile, issuer: domain }
} catch (error) {
console.error({ error })
}
}
function extractDomain({ hostname }) {
return hostname.replace(/.*?\.([^.]+\.[^.]+)$/, '$1')
}
/**
* Generates a JWT
* @param {Object} query
* @param {string|undefined} query.secret The secret used to encode and verify the JWT
* @param {string|undefined} query.issuer The identity of the JWT issuer
* @param {string|number|undefined} query.expirationTTL The JWT expiration timestamp as a number or a timespan string
* @param {Object|undefined} query.claims Additional claims to include in the JWT payload
* @returns A JWT generated from the query
* @throws The JWT could not be generated from the query
*/
async function generate({ secret, issuer, expirationTTL, audience, ...claims }) {
let signJwt = new SignJWT({ ...claims }).setProtectedHeader({ alg: 'HS256' }).setJti(nanoid()).setIssuedAt()
if (issuer) signJwt = signJwt.setIssuer(issuer)
if (audience) signJwt = signJwt.setAudience(audience)
if (expirationTTL) signJwt = signJwt.setExpirationTime(expirationTTL.match(/^\d+$/) ? parseInt(expirationTTL) : expirationTTL)
return await signJwt.sign(new Uint8Array(await crypto.subtle.digest('SHA-512', new TextEncoder().encode(secret.replaceAll(' ', '+')))))
}
/**
* Verifies a JWT
* @param {Object} query
* @param {string} query.token The JWT to be verified
* @param {string|undefined} query.secret The secret used to encode and verify the JWT
* @param {string|undefined} query.issuer The issuer of the JWT
* @returns The decoded payload and header
* @throws The JWT is not valid
*/
async function verify({ token, secret, issuer }) {
const hash = await crypto.subtle.digest('SHA-512', new TextEncoder().encode(secret.replaceAll(' ', '+')))
return await jwtVerify(token, new Uint8Array(hash), { issuer })
}