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.

318 lines 22.1 kB
import * as plugins from '../../plugins.js'; import { logger } from '../../core/utils/logger.js'; import { cleanupSocket } from '../../core/utils/socket-utils.js'; /** * PortManager handles the dynamic creation and removal of port listeners * * This class provides methods to add and remove listening ports at runtime, * allowing SmartProxy to adapt to configuration changes without requiring * a full restart. * * It includes a reference counting system to track how many routes are using * each port, so ports can be automatically released when they are no longer needed. */ export class PortManager { /** * Create a new PortManager * * @param smartProxy The SmartProxy instance */ constructor(smartProxy) { this.smartProxy = smartProxy; this.servers = new Map(); this.isShuttingDown = false; // Track how many routes are using each port this.portRefCounts = new Map(); } /** * Start listening on a specific port * * @param port The port number to listen on * @returns Promise that resolves when the server is listening or rejects on error */ async addPort(port) { // Check if we're already listening on this port if (this.servers.has(port)) { // Port is already bound, just increment the reference count this.incrementPortRefCount(port); try { logger.log('debug', `PortManager: Port ${port} is already bound by SmartProxy, reusing binding`, { port, component: 'port-manager' }); } catch (e) { console.log(`[DEBUG] PortManager: Port ${port} is already bound by SmartProxy, reusing binding`); } return; } // Initialize reference count for new port this.portRefCounts.set(port, 1); // Create a server for this port const server = plugins.net.createServer((socket) => { // Check if shutting down if (this.isShuttingDown) { cleanupSocket(socket, 'port-manager-shutdown', { immediate: true }); return; } // Delegate to route connection handler this.smartProxy.routeConnectionHandler.handleConnection(socket); }).on('error', (err) => { try { logger.log('error', `Server Error on port ${port}: ${err.message}`, { port, error: err.message, component: 'port-manager' }); } catch (e) { console.error(`[ERROR] Server Error on port ${port}: ${err.message}`); } }); // Start listening on the port return new Promise((resolve, reject) => { server.listen(port, () => { const isHttpProxyPort = this.smartProxy.settings.useHttpProxy?.includes(port); try { logger.log('info', `SmartProxy -> OK: Now listening on port ${port}${isHttpProxyPort ? ' (HttpProxy forwarding enabled)' : ''}`, { port, isHttpProxyPort: !!isHttpProxyPort, component: 'port-manager' }); } catch (e) { console.log(`[INFO] SmartProxy -> OK: Now listening on port ${port}${isHttpProxyPort ? ' (HttpProxy forwarding enabled)' : ''}`); } // Store the server reference this.servers.set(port, server); resolve(); }).on('error', (err) => { // Check if this is an external conflict const { isConflict, isExternal } = this.isPortConflict(err); if (isConflict && !isExternal) { // This is an internal conflict (port already bound by SmartProxy) // This shouldn't normally happen because we check servers.has(port) above logger.log('warn', `Port ${port} binding conflict: already in use by SmartProxy`, { port, component: 'port-manager' }); // Still increment reference count to maintain tracking this.incrementPortRefCount(port); resolve(); return; } // Log the error and propagate it logger.log('error', `Failed to listen on port ${port}: ${err.message}`, { port, error: err.message, code: err.code, component: 'port-manager' }); // Clean up reference count since binding failed this.portRefCounts.delete(port); reject(err); }); }); } /** * Stop listening on a specific port * * @param port The port to stop listening on * @returns Promise that resolves when the server is closed */ async removePort(port) { // Decrement the reference count first const newRefCount = this.decrementPortRefCount(port); // If there are still references to this port, keep it open if (newRefCount > 0) { logger.log('debug', `PortManager: Port ${port} still has ${newRefCount} references, keeping open`, { port, refCount: newRefCount, component: 'port-manager' }); return; } // Get the server for this port const server = this.servers.get(port); if (!server) { logger.log('warn', `PortManager: Not listening on port ${port}`, { port, component: 'port-manager' }); // Ensure reference count is reset this.portRefCounts.delete(port); return; } // Close the server return new Promise((resolve) => { server.close((err) => { if (err) { logger.log('error', `Error closing server on port ${port}: ${err.message}`, { port, error: err.message, component: 'port-manager' }); } else { logger.log('info', `SmartProxy -> Stopped listening on port ${port}`, { port, component: 'port-manager' }); } // Remove the server reference and clean up reference counting this.servers.delete(port); this.portRefCounts.delete(port); resolve(); }); }); } /** * Add multiple ports at once * * @param ports Array of ports to add * @returns Promise that resolves when all servers are listening */ async addPorts(ports) { const uniquePorts = [...new Set(ports)]; await Promise.all(uniquePorts.map(port => this.addPort(port))); } /** * Remove multiple ports at once * * @param ports Array of ports to remove * @returns Promise that resolves when all servers are closed */ async removePorts(ports) { const uniquePorts = [...new Set(ports)]; await Promise.all(uniquePorts.map(port => this.removePort(port))); } /** * Update listening ports to match the provided list * * This will add any ports that aren't currently listening, * and remove any ports that are no longer needed. * * @param ports Array of ports that should be listening * @returns Promise that resolves when all operations are complete */ async updatePorts(ports) { const targetPorts = new Set(ports); const currentPorts = new Set(this.servers.keys()); // Find ports to add and remove const portsToAdd = ports.filter(port => !currentPorts.has(port)); const portsToRemove = Array.from(currentPorts).filter(port => !targetPorts.has(port)); // Log the changes if (portsToAdd.length > 0) { console.log(`PortManager: Adding new listeners for ports: ${portsToAdd.join(', ')}`); } if (portsToRemove.length > 0) { console.log(`PortManager: Removing listeners for ports: ${portsToRemove.join(', ')}`); } // Add and remove ports await this.removePorts(portsToRemove); await this.addPorts(portsToAdd); } /** * Get all ports that are currently listening * * @returns Array of port numbers */ getListeningPorts() { return Array.from(this.servers.keys()); } /** * Mark the port manager as shutting down */ setShuttingDown(isShuttingDown) { this.isShuttingDown = isShuttingDown; } /** * Close all listening servers * * @returns Promise that resolves when all servers are closed */ async closeAll() { const allPorts = Array.from(this.servers.keys()); await this.removePorts(allPorts); } /** * Get all server instances (for testing or debugging) */ getServers() { return new Map(this.servers); } /** * Check if a port is bound by this SmartProxy instance * * @param port The port number to check * @returns True if the port is currently bound by SmartProxy */ isPortBoundBySmartProxy(port) { return this.servers.has(port); } /** * Get the current reference count for a port * * @param port The port number to check * @returns The number of routes using this port, 0 if none */ getPortRefCount(port) { return this.portRefCounts.get(port) || 0; } /** * Increment the reference count for a port * * @param port The port number to increment * @returns The new reference count */ incrementPortRefCount(port) { const currentCount = this.portRefCounts.get(port) || 0; const newCount = currentCount + 1; this.portRefCounts.set(port, newCount); logger.log('debug', `Port ${port} reference count increased to ${newCount}`, { port, refCount: newCount, component: 'port-manager' }); return newCount; } /** * Decrement the reference count for a port * * @param port The port number to decrement * @returns The new reference count */ decrementPortRefCount(port) { const currentCount = this.portRefCounts.get(port) || 0; if (currentCount <= 0) { logger.log('warn', `Attempted to decrement reference count for port ${port} below zero`, { port, component: 'port-manager' }); return 0; } const newCount = currentCount - 1; this.portRefCounts.set(port, newCount); logger.log('debug', `Port ${port} reference count decreased to ${newCount}`, { port, refCount: newCount, component: 'port-manager' }); return newCount; } /** * Determine if a port binding error is due to an external or internal conflict * * @param error The error object from a failed port binding * @returns Object indicating if this is a conflict and if it's external */ isPortConflict(error) { if (error.code !== 'EADDRINUSE') { return { isConflict: false, isExternal: false }; } // Check if we already have this port const isBoundInternally = this.servers.has(Number(error.port)); return { isConflict: true, isExternal: !isBoundInternally }; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"port-manager.js","sourceRoot":"","sources":["../../../ts/proxies/smart-proxy/port-manager.ts"],"names":[],"mappings":"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"}