| 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; |