Files
pnpm/network/fetch/src/fetchFromRegistry.ts

101 lines
3.0 KiB
TypeScript

import { URL } from 'url'
import { FetchFromRegistry } from '@pnpm/fetching-types'
import { getAgent, AgentOptions } from '@pnpm/network.agent'
import { fetch, isRedirect, Response, RequestInfo, RequestInit } from './fetch'
const USER_AGENT = 'pnpm' // or maybe make it `${pkg.name}/${pkg.version} (+https://npm.im/${pkg.name})`
const ABBREVIATED_DOC = 'application/vnd.npm.install-v1+json; q=1.0, application/json; q=0.8, */*'
const JSON_DOC = 'application/json'
const MAX_FOLLOWED_REDIRECTS = 20
export type FetchWithAgentOptions = RequestInit & {
agentOptions: AgentOptions
}
export function fetchWithAgent (url: RequestInfo, opts: FetchWithAgentOptions) {
const agent = getAgent(url.toString(), {
...opts.agentOptions,
strictSsl: opts.agentOptions.strictSsl ?? true,
} as any) as any // eslint-disable-line
const headers = opts.headers ?? {}
// @ts-expect-error
headers['connection'] = agent ? 'keep-alive' : 'close'
return fetch(url, {
...opts,
agent,
})
}
export { AgentOptions }
export function createFetchFromRegistry (
defaultOpts: {
fullMetadata?: boolean
userAgent?: string
} & AgentOptions
): FetchFromRegistry {
return async (url, opts): Promise<Response> => {
const headers = {
'user-agent': USER_AGENT,
...getHeaders({
auth: opts?.authHeaderValue,
fullMetadata: defaultOpts.fullMetadata,
userAgent: defaultOpts.userAgent,
}),
}
let redirects = 0
let urlObject = new URL(url)
const originalHost = urlObject.host
while (true) {
const agentOptions = {
...defaultOpts,
...opts,
strictSsl: defaultOpts.strictSsl ?? true,
} as any // eslint-disable-line
// We should pass a URL object to node-fetch till this is not resolved:
// https://github.com/bitinn/node-fetch/issues/245
const response = await fetchWithAgent(urlObject, {
agentOptions,
// if verifying integrity, node-fetch must not decompress
compress: opts?.compress ?? false,
headers,
redirect: 'manual',
retry: opts?.retry,
timeout: opts?.timeout ?? 60000,
})
if (!isRedirect(response.status) || redirects >= MAX_FOLLOWED_REDIRECTS) {
return response
}
// This is a workaround to remove authorization headers on redirect.
// Related pnpm issue: https://github.com/pnpm/pnpm/issues/1815
redirects++
urlObject = new URL(response.headers.get('location')!)
if (!headers['authorization'] || originalHost === urlObject.host) continue
delete headers.authorization
}
}
}
function getHeaders (
opts: {
auth?: string
fullMetadata?: boolean
userAgent?: string
}
) {
const headers: { accept: string, authorization?: string, 'user-agent'?: string } = {
accept: opts.fullMetadata === true ? JSON_DOC : ABBREVIATED_DOC,
}
if (opts.auth) {
headers['authorization'] = opts.auth
}
if (opts.userAgent) {
headers['user-agent'] = opts.userAgent
}
return headers
}