Spaces:
Sleeping
Sleeping
| import net from 'net'; | |
| import http from 'http'; | |
| import https from 'https'; | |
| import { Duplex } from 'stream'; | |
| import { EventEmitter } from 'events'; | |
| import createDebug from 'debug'; | |
| import promisify from './promisify'; | |
| const debug = createDebug('agent-base'); | |
| function isAgent(v: any): v is createAgent.AgentLike { | |
| return Boolean(v) && typeof v.addRequest === 'function'; | |
| } | |
| function isSecureEndpoint(): boolean { | |
| const { stack } = new Error(); | |
| if (typeof stack !== 'string') return false; | |
| return stack.split('\n').some(l => l.indexOf('(https.js:') !== -1 || l.indexOf('node:https:') !== -1); | |
| } | |
| function createAgent(opts?: createAgent.AgentOptions): createAgent.Agent; | |
| function createAgent( | |
| callback: createAgent.AgentCallback, | |
| opts?: createAgent.AgentOptions | |
| ): createAgent.Agent; | |
| function createAgent( | |
| callback?: createAgent.AgentCallback | createAgent.AgentOptions, | |
| opts?: createAgent.AgentOptions | |
| ) { | |
| return new createAgent.Agent(callback, opts); | |
| } | |
| namespace createAgent { | |
| export interface ClientRequest extends http.ClientRequest { | |
| _last?: boolean; | |
| _hadError?: boolean; | |
| method: string; | |
| } | |
| export interface AgentRequestOptions { | |
| host?: string; | |
| path?: string; | |
| // `port` on `http.RequestOptions` can be a string or undefined, | |
| // but `net.TcpNetConnectOpts` expects only a number | |
| port: number; | |
| } | |
| export interface HttpRequestOptions | |
| extends AgentRequestOptions, | |
| Omit<http.RequestOptions, keyof AgentRequestOptions> { | |
| secureEndpoint: false; | |
| } | |
| export interface HttpsRequestOptions | |
| extends AgentRequestOptions, | |
| Omit<https.RequestOptions, keyof AgentRequestOptions> { | |
| secureEndpoint: true; | |
| } | |
| export type RequestOptions = HttpRequestOptions | HttpsRequestOptions; | |
| export type AgentLike = Pick<createAgent.Agent, 'addRequest'> | http.Agent; | |
| export type AgentCallbackReturn = Duplex | AgentLike; | |
| export type AgentCallbackCallback = ( | |
| err?: Error | null, | |
| socket?: createAgent.AgentCallbackReturn | |
| ) => void; | |
| export type AgentCallbackPromise = ( | |
| req: createAgent.ClientRequest, | |
| opts: createAgent.RequestOptions | |
| ) => | |
| | createAgent.AgentCallbackReturn | |
| | Promise<createAgent.AgentCallbackReturn>; | |
| export type AgentCallback = typeof Agent.prototype.callback; | |
| export type AgentOptions = { | |
| timeout?: number; | |
| }; | |
| /** | |
| * Base `http.Agent` implementation. | |
| * No pooling/keep-alive is implemented by default. | |
| * | |
| * @param {Function} callback | |
| * @api public | |
| */ | |
| export class Agent extends EventEmitter { | |
| public timeout: number | null; | |
| public maxFreeSockets: number; | |
| public maxTotalSockets: number; | |
| public maxSockets: number; | |
| public sockets: { | |
| [key: string]: net.Socket[]; | |
| }; | |
| public freeSockets: { | |
| [key: string]: net.Socket[]; | |
| }; | |
| public requests: { | |
| [key: string]: http.IncomingMessage[]; | |
| }; | |
| public options: https.AgentOptions; | |
| private promisifiedCallback?: createAgent.AgentCallbackPromise; | |
| private explicitDefaultPort?: number; | |
| private explicitProtocol?: string; | |
| constructor( | |
| callback?: createAgent.AgentCallback | createAgent.AgentOptions, | |
| _opts?: createAgent.AgentOptions | |
| ) { | |
| super(); | |
| let opts = _opts; | |
| if (typeof callback === 'function') { | |
| this.callback = callback; | |
| } else if (callback) { | |
| opts = callback; | |
| } | |
| // Timeout for the socket to be returned from the callback | |
| this.timeout = null; | |
| if (opts && typeof opts.timeout === 'number') { | |
| this.timeout = opts.timeout; | |
| } | |
| // These aren't actually used by `agent-base`, but are required | |
| // for the TypeScript definition files in `@types/node` :/ | |
| this.maxFreeSockets = 1; | |
| this.maxSockets = 1; | |
| this.maxTotalSockets = Infinity; | |
| this.sockets = {}; | |
| this.freeSockets = {}; | |
| this.requests = {}; | |
| this.options = {}; | |
| } | |
| get defaultPort(): number { | |
| if (typeof this.explicitDefaultPort === 'number') { | |
| return this.explicitDefaultPort; | |
| } | |
| return isSecureEndpoint() ? 443 : 80; | |
| } | |
| set defaultPort(v: number) { | |
| this.explicitDefaultPort = v; | |
| } | |
| get protocol(): string { | |
| if (typeof this.explicitProtocol === 'string') { | |
| return this.explicitProtocol; | |
| } | |
| return isSecureEndpoint() ? 'https:' : 'http:'; | |
| } | |
| set protocol(v: string) { | |
| this.explicitProtocol = v; | |
| } | |
| callback( | |
| req: createAgent.ClientRequest, | |
| opts: createAgent.RequestOptions, | |
| fn: createAgent.AgentCallbackCallback | |
| ): void; | |
| callback( | |
| req: createAgent.ClientRequest, | |
| opts: createAgent.RequestOptions | |
| ): | |
| | createAgent.AgentCallbackReturn | |
| | Promise<createAgent.AgentCallbackReturn>; | |
| callback( | |
| req: createAgent.ClientRequest, | |
| opts: createAgent.AgentOptions, | |
| fn?: createAgent.AgentCallbackCallback | |
| ): | |
| | createAgent.AgentCallbackReturn | |
| | Promise<createAgent.AgentCallbackReturn> | |
| | void { | |
| throw new Error( | |
| '"agent-base" has no default implementation, you must subclass and override `callback()`' | |
| ); | |
| } | |
| /** | |
| * Called by node-core's "_http_client.js" module when creating | |
| * a new HTTP request with this Agent instance. | |
| * | |
| * @api public | |
| */ | |
| addRequest(req: ClientRequest, _opts: RequestOptions): void { | |
| const opts: RequestOptions = { ..._opts }; | |
| if (typeof opts.secureEndpoint !== 'boolean') { | |
| opts.secureEndpoint = isSecureEndpoint(); | |
| } | |
| if (opts.host == null) { | |
| opts.host = 'localhost'; | |
| } | |
| if (opts.port == null) { | |
| opts.port = opts.secureEndpoint ? 443 : 80; | |
| } | |
| if (opts.protocol == null) { | |
| opts.protocol = opts.secureEndpoint ? 'https:' : 'http:'; | |
| } | |
| if (opts.host && opts.path) { | |
| // If both a `host` and `path` are specified then it's most | |
| // likely the result of a `url.parse()` call... we need to | |
| // remove the `path` portion so that `net.connect()` doesn't | |
| // attempt to open that as a unix socket file. | |
| delete opts.path; | |
| } | |
| delete opts.agent; | |
| delete opts.hostname; | |
| delete opts._defaultAgent; | |
| delete opts.defaultPort; | |
| delete opts.createConnection; | |
| // Hint to use "Connection: close" | |
| // XXX: non-documented `http` module API :( | |
| req._last = true; | |
| req.shouldKeepAlive = false; | |
| let timedOut = false; | |
| let timeoutId: ReturnType<typeof setTimeout> | null = null; | |
| const timeoutMs = opts.timeout || this.timeout; | |
| const onerror = (err: NodeJS.ErrnoException) => { | |
| if (req._hadError) return; | |
| req.emit('error', err); | |
| // For Safety. Some additional errors might fire later on | |
| // and we need to make sure we don't double-fire the error event. | |
| req._hadError = true; | |
| }; | |
| const ontimeout = () => { | |
| timeoutId = null; | |
| timedOut = true; | |
| const err: NodeJS.ErrnoException = new Error( | |
| `A "socket" was not created for HTTP request before ${timeoutMs}ms` | |
| ); | |
| err.code = 'ETIMEOUT'; | |
| onerror(err); | |
| }; | |
| const callbackError = (err: NodeJS.ErrnoException) => { | |
| if (timedOut) return; | |
| if (timeoutId !== null) { | |
| clearTimeout(timeoutId); | |
| timeoutId = null; | |
| } | |
| onerror(err); | |
| }; | |
| const onsocket = (socket: AgentCallbackReturn) => { | |
| if (timedOut) return; | |
| if (timeoutId != null) { | |
| clearTimeout(timeoutId); | |
| timeoutId = null; | |
| } | |
| if (isAgent(socket)) { | |
| // `socket` is actually an `http.Agent` instance, so | |
| // relinquish responsibility for this `req` to the Agent | |
| // from here on | |
| debug( | |
| 'Callback returned another Agent instance %o', | |
| socket.constructor.name | |
| ); | |
| (socket as createAgent.Agent).addRequest(req, opts); | |
| return; | |
| } | |
| if (socket) { | |
| socket.once('free', () => { | |
| this.freeSocket(socket as net.Socket, opts); | |
| }); | |
| req.onSocket(socket as net.Socket); | |
| return; | |
| } | |
| const err = new Error( | |
| `no Duplex stream was returned to agent-base for \`${req.method} ${req.path}\`` | |
| ); | |
| onerror(err); | |
| }; | |
| if (typeof this.callback !== 'function') { | |
| onerror(new Error('`callback` is not defined')); | |
| return; | |
| } | |
| if (!this.promisifiedCallback) { | |
| if (this.callback.length >= 3) { | |
| debug('Converting legacy callback function to promise'); | |
| this.promisifiedCallback = promisify(this.callback); | |
| } else { | |
| this.promisifiedCallback = this.callback; | |
| } | |
| } | |
| if (typeof timeoutMs === 'number' && timeoutMs > 0) { | |
| timeoutId = setTimeout(ontimeout, timeoutMs); | |
| } | |
| if ('port' in opts && typeof opts.port !== 'number') { | |
| opts.port = Number(opts.port); | |
| } | |
| try { | |
| debug( | |
| 'Resolving socket for %o request: %o', | |
| opts.protocol, | |
| `${req.method} ${req.path}` | |
| ); | |
| Promise.resolve(this.promisifiedCallback(req, opts)).then( | |
| onsocket, | |
| callbackError | |
| ); | |
| } catch (err) { | |
| Promise.reject(err).catch(callbackError); | |
| } | |
| } | |
| freeSocket(socket: net.Socket, opts: AgentOptions) { | |
| debug('Freeing socket %o %o', socket.constructor.name, opts); | |
| socket.destroy(); | |
| } | |
| destroy() { | |
| debug('Destroying agent %o', this.constructor.name); | |
| } | |
| } | |
| // So that `instanceof` works correctly | |
| createAgent.prototype = createAgent.Agent.prototype; | |
| } | |
| export = createAgent; | |