mirror of
https://github.com/danbulant/jose
synced 2026-05-19 04:18:52 +00:00
63 lines
1.5 KiB
TypeScript
63 lines
1.5 KiB
TypeScript
import * as http from 'http'
|
|
import * as https from 'https'
|
|
import { once } from 'events'
|
|
import type { ClientRequest, IncomingMessage } from 'http'
|
|
import type { RequestOptions } from 'https'
|
|
|
|
import type { FetchFunction } from '../interfaces.d'
|
|
import { JOSEError, JWKSTimeout } from '../../util/errors.js'
|
|
import { concat, decoder } from '../../lib/buffer_utils.js'
|
|
|
|
type AcceptedRequestOptions = Pick<RequestOptions, 'agent' | 'headers'>
|
|
|
|
const fetchJwks: FetchFunction = async (
|
|
url: URL,
|
|
timeout: number,
|
|
options: AcceptedRequestOptions,
|
|
) => {
|
|
let get: (...args: Parameters<typeof https.get>) => ClientRequest
|
|
switch (url.protocol) {
|
|
case 'https:':
|
|
get = https.get
|
|
break
|
|
case 'http:':
|
|
get = http.get
|
|
break
|
|
default:
|
|
throw new TypeError('Unsupported URL protocol.')
|
|
}
|
|
|
|
const { agent, headers } = options
|
|
const req = get(url.href, {
|
|
agent,
|
|
timeout,
|
|
headers,
|
|
})
|
|
|
|
const [response] = <[IncomingMessage]>(
|
|
await Promise.race([once(req, 'response'), once(req, 'timeout')])
|
|
)
|
|
|
|
// timeout reached
|
|
if (!response) {
|
|
req.destroy()
|
|
throw new JWKSTimeout()
|
|
}
|
|
|
|
if (response.statusCode !== 200) {
|
|
throw new JOSEError('Expected 200 OK from the JSON Web Key Set HTTP response')
|
|
}
|
|
|
|
const parts = []
|
|
for await (const part of response) {
|
|
parts.push(part)
|
|
}
|
|
|
|
try {
|
|
return JSON.parse(decoder.decode(concat(...parts)))
|
|
} catch {
|
|
throw new JOSEError('Failed to parse the JSON Web Key Set HTTP response as JSON')
|
|
}
|
|
}
|
|
|
|
export default fetchJwks
|