UNPKG

@push.rocks/smartproxy

Version:

A powerful proxy package with unified route-based configuration for high traffic management. Features include SSL/TLS support, flexible routing patterns, WebSocket handling, advanced security options, and automatic ACME certificate management.

201 lines 18.6 kB
import * as plugins from '../../plugins.js'; /** * HTTP/2 Request Handler Helper - handles HTTP/2 streams with specific destinations * This is a helper class for the main RequestHandler */ export class Http2RequestHandler { /** * Handle HTTP/2 stream with direct HTTP/2 backend */ static async handleHttp2WithHttp2Destination(stream, headers, destination, routeContext, sessions, logger, metricsTracker) { const key = `${destination.host}:${destination.port}`; // Get or create a client HTTP/2 session let session = sessions.get(key); if (!session || session.closed || session.destroyed) { try { // Connect to the backend HTTP/2 server session = plugins.http2.connect(`http://${destination.host}:${destination.port}`); sessions.set(key, session); // Handle session errors and cleanup session.on('error', (err) => { logger.error(`HTTP/2 session error to ${key}: ${err.message}`); sessions.delete(key); }); session.on('close', () => { logger.debug(`HTTP/2 session closed to ${key}`); sessions.delete(key); }); } catch (err) { logger.error(`Failed to establish HTTP/2 session to ${key}: ${err.message}`); stream.respond({ ':status': 502 }); stream.end('Bad Gateway: Failed to establish connection to backend'); if (metricsTracker) metricsTracker.incrementFailedRequests(); return; } } try { // Build headers for backend HTTP/2 request const h2Headers = { ':method': headers[':method'], ':path': headers[':path'], ':authority': `${destination.host}:${destination.port}` }; // Copy other headers, excluding pseudo-headers for (const [key, value] of Object.entries(headers)) { if (!key.startsWith(':') && typeof value === 'string') { h2Headers[key] = value; } } logger.debug(`Proxying HTTP/2 request to ${destination.host}:${destination.port}${headers[':path']}`, { method: headers[':method'] }); // Create HTTP/2 request stream to the backend const h2Stream = session.request(h2Headers); // Pipe client stream to backend stream stream.pipe(h2Stream); // Handle responses from the backend h2Stream.on('response', (responseHeaders) => { // Map status and headers to client response const resp = { ':status': responseHeaders[':status'] }; // Copy non-pseudo headers for (const [key, value] of Object.entries(responseHeaders)) { if (!key.startsWith(':') && value !== undefined) { resp[key] = value; } } // Send headers to client stream.respond(resp); // Pipe backend response to client h2Stream.pipe(stream); // Track successful requests stream.on('end', () => { if (metricsTracker) metricsTracker.incrementRequestsServed(); logger.debug(`HTTP/2 request completed: ${headers[':method']} ${headers[':path']} ${responseHeaders[':status']}`, { method: headers[':method'], status: responseHeaders[':status'] }); }); }); // Handle backend errors h2Stream.on('error', (err) => { logger.error(`HTTP/2 stream error: ${err.message}`); // Only send error response if headers haven't been sent if (!stream.headersSent) { stream.respond({ ':status': 502 }); stream.end(`Bad Gateway: ${err.message}`); } else { stream.end(); } if (metricsTracker) metricsTracker.incrementFailedRequests(); }); // Handle client stream errors stream.on('error', (err) => { logger.debug(`Client HTTP/2 stream error: ${err.message}`); h2Stream.destroy(); if (metricsTracker) metricsTracker.incrementFailedRequests(); }); } catch (err) { logger.error(`Error handling HTTP/2 request: ${err.message}`); // Only send error response if headers haven't been sent if (!stream.headersSent) { stream.respond({ ':status': 500 }); stream.end('Internal Server Error'); } else { stream.end(); } if (metricsTracker) metricsTracker.incrementFailedRequests(); } } /** * Handle HTTP/2 stream with HTTP/1 backend */ static async handleHttp2WithHttp1Destination(stream, headers, destination, routeContext, logger, metricsTracker) { try { // Build headers for HTTP/1 proxy request, excluding HTTP/2 pseudo-headers const outboundHeaders = {}; for (const [key, value] of Object.entries(headers)) { if (typeof key === 'string' && typeof value === 'string' && !key.startsWith(':')) { outboundHeaders[key] = value; } } // Always rewrite host header to match target outboundHeaders.host = `${destination.host}:${destination.port}`; logger.debug(`Proxying HTTP/2 request to HTTP/1 backend ${destination.host}:${destination.port}${headers[':path']}`, { method: headers[':method'] }); // Create HTTP/1 proxy request const proxyReq = plugins.http.request({ hostname: destination.host, port: destination.port, path: headers[':path'], method: headers[':method'], headers: outboundHeaders }, (proxyRes) => { // Map status and headers back to HTTP/2 const responseHeaders = { ':status': proxyRes.statusCode || 500 }; // Copy headers from HTTP/1 response to HTTP/2 response for (const [key, value] of Object.entries(proxyRes.headers)) { if (value !== undefined) { responseHeaders[key] = value; } } // Send headers to client stream.respond(responseHeaders); // Pipe HTTP/1 response to HTTP/2 stream proxyRes.pipe(stream); // Clean up when client disconnects stream.on('close', () => proxyReq.destroy()); stream.on('error', () => proxyReq.destroy()); // Track successful requests stream.on('end', () => { if (metricsTracker) metricsTracker.incrementRequestsServed(); logger.debug(`HTTP/2 to HTTP/1 request completed: ${headers[':method']} ${headers[':path']} ${proxyRes.statusCode}`, { method: headers[':method'], status: proxyRes.statusCode }); }); }); // Handle proxy request errors proxyReq.on('error', (err) => { logger.error(`HTTP/1 proxy error: ${err.message}`); // Only send error response if headers haven't been sent if (!stream.headersSent) { stream.respond({ ':status': 502 }); stream.end(`Bad Gateway: ${err.message}`); } else { stream.end(); } if (metricsTracker) metricsTracker.incrementFailedRequests(); }); // Pipe client stream to proxy request stream.pipe(proxyReq); // Handle client stream errors stream.on('error', (err) => { logger.debug(`Client HTTP/2 stream error: ${err.message}`); proxyReq.destroy(); if (metricsTracker) metricsTracker.incrementFailedRequests(); }); } catch (err) { logger.error(`Error handling HTTP/2 to HTTP/1 request: ${err.message}`); // Only send error response if headers haven't been sent if (!stream.headersSent) { stream.respond({ ':status': 500 }); stream.end('Internal Server Error'); } else { stream.end(); } if (metricsTracker) metricsTracker.incrementFailedRequests(); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http2-request-handler.js","sourceRoot":"","sources":["../../../ts/proxies/http-proxy/http2-request-handler.ts"],"names":[],"mappings":"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"}