n8n/packages/@n8n/nodes-langchain/utils/httpProxyAgent.ts

107 lines
4.2 KiB
TypeScript

import { HttpsProxyAgent } from 'https-proxy-agent';
import proxyFromEnv from 'proxy-from-env';
import { Agent, ProxyAgent } from 'undici';
/**
* Options for configuring HTTP agent timeouts.
* These timeouts are passed to undici's Agent/ProxyAgent to override default 5-minute timeouts.
*/
export interface AgentTimeoutOptions {
headersTimeout?: number;
bodyTimeout?: number;
connectTimeout?: number;
}
// Default timeout for AI operations (1 hour)
// Aligned with EXECUTIONS_TIMEOUT_MAX to ensure AI requests don't exceed workflow execution limits
// Configurable via N8N_AI_TIMEOUT_MAX environment variable to support custom timeout requirements
const DEFAULT_TIMEOUT = parseInt(process.env.N8N_AI_TIMEOUT_MAX ?? '3600000', 10);
/**
* Resolves the proxy URL from environment variables for a given target URL.
*
* @param targetUrl - The target URL to check proxy configuration for (optional)
* @returns The proxy URL string or undefined if no proxy is configured
*
* @remarks
* There are cases where we don't know the target URL in advance (e.g. when we need to provide a proxy agent to ChatAwsBedrock).
* In such case we use a dummy URL.
* This will lead to `NO_PROXY` environment variable not being respected, but it is better than not having a proxy agent at all.
*/
function getProxyUrlFromEnv(targetUrl?: string): string {
return proxyFromEnv.getProxyForUrl(targetUrl ?? 'https://example.nonexistent/');
}
/**
* Returns an undici Agent or ProxyAgent with configured timeouts based on the environment variables and target URL.
* When target URL is not provided, NO_PROXY environment variable is not respected.
*
* @param targetUrl - The target URL to check proxy configuration for (optional)
* @param timeoutOptions - Optional timeout configuration to override defaults. When provided,
* always returns an Agent/ProxyAgent (even without proxy) to ensure timeouts are applied.
* @returns An Agent (no proxy with timeout options) or ProxyAgent (with proxy) configured with timeouts,
* or undefined if no proxy is configured and no timeout options are provided (backward compatible behavior).
*
* @remarks
* When timeoutOptions are provided, this function always returns an agent to ensure timeouts are properly configured.
* The default undici timeouts (5 minutes) are too short for many AI operations.
* When timeoutOptions are NOT provided, returns undefined if no proxy is configured (backward compatible).
*/
export function getProxyAgent(targetUrl?: string, timeoutOptions?: AgentTimeoutOptions) {
const proxyUrl = getProxyUrlFromEnv(targetUrl);
const agentOptions = {
headersTimeout: timeoutOptions?.headersTimeout ?? DEFAULT_TIMEOUT,
bodyTimeout: timeoutOptions?.bodyTimeout ?? DEFAULT_TIMEOUT,
...(timeoutOptions?.connectTimeout !== undefined && {
connectTimeout: timeoutOptions.connectTimeout,
}),
};
if (!proxyUrl) {
if (timeoutOptions) {
return new Agent(agentOptions);
}
return undefined;
}
return new ProxyAgent({ uri: proxyUrl, ...agentOptions });
}
/**
* Make a fetch() request with an Agent/ProxyAgent that has configured timeouts.
* If proxy environment variables are set, uses ProxyAgent; otherwise uses Agent.
*
* @param input - The URL to fetch
* @param init - Standard fetch RequestInit options
* @param timeoutOptions - Optional timeout configuration to override defaults
*/
export async function proxyFetch(
input: string | URL,
init?: RequestInit,
timeoutOptions?: AgentTimeoutOptions,
): Promise<Response> {
return await fetch(input, {
...init,
// @ts-expect-error - dispatcher is an undici-specific option not in standard fetch
dispatcher: getProxyAgent(input.toString(), timeoutOptions),
});
}
/**
* Returns a Node.js HTTP/HTTPS proxy agent for use with AWS SDK v3 clients.
* AWS SDK v3 requires Node.js http.Agent/https.Agent instances (not undici ProxyAgent).
*
* @param targetUrl - The target URL to check proxy configuration for
* @returns HttpsProxyAgent instance or undefined if no proxy is configured
*/
export function getNodeProxyAgent(targetUrl?: string) {
const proxyUrl = getProxyUrlFromEnv(targetUrl);
if (!proxyUrl) {
return undefined;
}
return new HttpsProxyAgent(proxyUrl);
}