278 lines
8.1 KiB
TypeScript
278 lines
8.1 KiB
TypeScript
import * as plugins from '../plugins.js';
|
|
import { type INetworkProxyOptions, type ILogger, createLogger, type IReverseProxyConfig } from './classes.np.types.js';
|
|
import { ConnectionPool } from './classes.np.connectionpool.js';
|
|
import { ProxyRouter } from '../classes.router.js';
|
|
|
|
/**
|
|
* Interface for tracking metrics
|
|
*/
|
|
export interface IMetricsTracker {
|
|
incrementRequestsServed(): void;
|
|
incrementFailedRequests(): void;
|
|
}
|
|
|
|
/**
|
|
* Handles HTTP request processing and proxying
|
|
*/
|
|
export class RequestHandler {
|
|
private defaultHeaders: { [key: string]: string } = {};
|
|
private logger: ILogger;
|
|
private metricsTracker: IMetricsTracker | null = null;
|
|
|
|
constructor(
|
|
private options: INetworkProxyOptions,
|
|
private connectionPool: ConnectionPool,
|
|
private router: ProxyRouter
|
|
) {
|
|
this.logger = createLogger(options.logLevel || 'info');
|
|
}
|
|
|
|
/**
|
|
* Set the metrics tracker instance
|
|
*/
|
|
public setMetricsTracker(tracker: IMetricsTracker): void {
|
|
this.metricsTracker = tracker;
|
|
}
|
|
|
|
/**
|
|
* Set default headers to be included in all responses
|
|
*/
|
|
public setDefaultHeaders(headers: { [key: string]: string }): void {
|
|
this.defaultHeaders = {
|
|
...this.defaultHeaders,
|
|
...headers
|
|
};
|
|
this.logger.info('Updated default response headers');
|
|
}
|
|
|
|
/**
|
|
* Get all default headers
|
|
*/
|
|
public getDefaultHeaders(): { [key: string]: string } {
|
|
return { ...this.defaultHeaders };
|
|
}
|
|
|
|
/**
|
|
* Apply CORS headers to response if configured
|
|
*/
|
|
private applyCorsHeaders(
|
|
res: plugins.http.ServerResponse,
|
|
req: plugins.http.IncomingMessage
|
|
): void {
|
|
if (!this.options.cors) {
|
|
return;
|
|
}
|
|
|
|
// Apply CORS headers
|
|
if (this.options.cors.allowOrigin) {
|
|
res.setHeader('Access-Control-Allow-Origin', this.options.cors.allowOrigin);
|
|
}
|
|
|
|
if (this.options.cors.allowMethods) {
|
|
res.setHeader('Access-Control-Allow-Methods', this.options.cors.allowMethods);
|
|
}
|
|
|
|
if (this.options.cors.allowHeaders) {
|
|
res.setHeader('Access-Control-Allow-Headers', this.options.cors.allowHeaders);
|
|
}
|
|
|
|
if (this.options.cors.maxAge) {
|
|
res.setHeader('Access-Control-Max-Age', this.options.cors.maxAge.toString());
|
|
}
|
|
|
|
// Handle CORS preflight requests
|
|
if (req.method === 'OPTIONS') {
|
|
res.statusCode = 204; // No content
|
|
res.end();
|
|
return;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Apply default headers to response
|
|
*/
|
|
private applyDefaultHeaders(res: plugins.http.ServerResponse): void {
|
|
// Apply default headers
|
|
for (const [key, value] of Object.entries(this.defaultHeaders)) {
|
|
if (!res.hasHeader(key)) {
|
|
res.setHeader(key, value);
|
|
}
|
|
}
|
|
|
|
// Add server identifier if not already set
|
|
if (!res.hasHeader('Server')) {
|
|
res.setHeader('Server', 'NetworkProxy');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handle an HTTP request
|
|
*/
|
|
public async handleRequest(
|
|
req: plugins.http.IncomingMessage,
|
|
res: plugins.http.ServerResponse
|
|
): Promise<void> {
|
|
// Record start time for logging
|
|
const startTime = Date.now();
|
|
|
|
// Apply CORS headers if configured
|
|
this.applyCorsHeaders(res, req);
|
|
|
|
// If this is an OPTIONS request, the response has already been ended in applyCorsHeaders
|
|
// so we should return early to avoid trying to set more headers
|
|
if (req.method === 'OPTIONS') {
|
|
// Increment metrics for OPTIONS requests too
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementRequestsServed();
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Apply default headers
|
|
this.applyDefaultHeaders(res);
|
|
|
|
try {
|
|
// Find target based on hostname
|
|
const proxyConfig = this.router.routeReq(req);
|
|
|
|
if (!proxyConfig) {
|
|
// No matching proxy configuration
|
|
this.logger.warn(`No proxy configuration for host: ${req.headers.host}`);
|
|
res.statusCode = 404;
|
|
res.end('Not Found: No proxy configuration for this host');
|
|
|
|
// Increment failed requests counter
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementFailedRequests();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// Get destination IP using round-robin if multiple IPs configured
|
|
const destination = this.connectionPool.getNextTarget(
|
|
proxyConfig.destinationIps,
|
|
proxyConfig.destinationPorts[0]
|
|
);
|
|
|
|
// Create options for the proxy request
|
|
const options: plugins.http.RequestOptions = {
|
|
hostname: destination.host,
|
|
port: destination.port,
|
|
path: req.url,
|
|
method: req.method,
|
|
headers: { ...req.headers }
|
|
};
|
|
|
|
// Remove host header to avoid issues with virtual hosts on target server
|
|
// The host header should match the target server's expected hostname
|
|
if (options.headers && options.headers.host) {
|
|
if ((proxyConfig as IReverseProxyConfig).rewriteHostHeader) {
|
|
options.headers.host = `${destination.host}:${destination.port}`;
|
|
}
|
|
}
|
|
|
|
this.logger.debug(
|
|
`Proxying request to ${destination.host}:${destination.port}${req.url}`,
|
|
{ method: req.method }
|
|
);
|
|
|
|
// Create proxy request
|
|
const proxyReq = plugins.http.request(options, (proxyRes) => {
|
|
// Copy status code
|
|
res.statusCode = proxyRes.statusCode || 500;
|
|
|
|
// Copy headers from proxy response to client response
|
|
for (const [key, value] of Object.entries(proxyRes.headers)) {
|
|
if (value !== undefined) {
|
|
res.setHeader(key, value);
|
|
}
|
|
}
|
|
|
|
// Pipe proxy response to client response
|
|
proxyRes.pipe(res);
|
|
|
|
// Increment served requests counter when the response finishes
|
|
res.on('finish', () => {
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementRequestsServed();
|
|
}
|
|
|
|
// Log the completed request
|
|
const duration = Date.now() - startTime;
|
|
this.logger.debug(
|
|
`Request completed in ${duration}ms: ${req.method} ${req.url} ${res.statusCode}`,
|
|
{ duration, statusCode: res.statusCode }
|
|
);
|
|
});
|
|
});
|
|
|
|
// Handle proxy request errors
|
|
proxyReq.on('error', (error) => {
|
|
const duration = Date.now() - startTime;
|
|
this.logger.error(
|
|
`Proxy error for ${req.method} ${req.url}: ${error.message}`,
|
|
{ duration, error: error.message }
|
|
);
|
|
|
|
// Increment failed requests counter
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementFailedRequests();
|
|
}
|
|
|
|
// Check if headers have already been sent
|
|
if (!res.headersSent) {
|
|
res.statusCode = 502;
|
|
res.end(`Bad Gateway: ${error.message}`);
|
|
} else {
|
|
// If headers already sent, just close the connection
|
|
res.end();
|
|
}
|
|
});
|
|
|
|
// Pipe request body to proxy request and handle client-side errors
|
|
req.pipe(proxyReq);
|
|
|
|
// Handle client disconnection
|
|
req.on('error', (error) => {
|
|
this.logger.debug(`Client connection error: ${error.message}`);
|
|
proxyReq.destroy();
|
|
|
|
// Increment failed requests counter on client errors
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementFailedRequests();
|
|
}
|
|
});
|
|
|
|
// Handle response errors
|
|
res.on('error', (error) => {
|
|
this.logger.debug(`Response error: ${error.message}`);
|
|
proxyReq.destroy();
|
|
|
|
// Increment failed requests counter on response errors
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementFailedRequests();
|
|
}
|
|
});
|
|
|
|
} catch (error) {
|
|
// Handle any unexpected errors
|
|
this.logger.error(
|
|
`Unexpected error handling request: ${error.message}`,
|
|
{ error: error.stack }
|
|
);
|
|
|
|
// Increment failed requests counter
|
|
if (this.metricsTracker) {
|
|
this.metricsTracker.incrementFailedRequests();
|
|
}
|
|
|
|
if (!res.headersSent) {
|
|
res.statusCode = 500;
|
|
res.end('Internal Server Error');
|
|
} else {
|
|
res.end();
|
|
}
|
|
}
|
|
}
|
|
} |