require('dotenv').config() const throng = require('throng') const os = require('os') const portUsed = require('port-used') const prefixStreamWrite = require('./lib/prefix-stream-write') // Intentionally require these for both cluster primary and workers require('./lib/check-node-version') require('./lib/handle-exceptions') require('./lib/feature-flags') const { PORT, NODE_ENV } = process.env const port = Number(PORT) || 4000 function main () { // Spin up a cluster! throng({ master: setupPrimary, worker: setupWorker, count: calculateWorkerCount() }) } // Start the server! main() // This function will only be run in the primary process async function setupPrimary () { process.on('beforeExit', () => { console.log('Shutting down primary...') console.log('Exiting!') }) console.log('Starting up primary...') // Check that the development server is not already running const portInUse = await portUsed.check(port) if (portInUse) { console.log(`\n\n\nPort ${port} is not available. You may already have a server running.`) console.log('Try running `killall node` to shut down all your running node processes.\n\n\n') console.log('\x07') // system 'beep' sound process.exit(1) } } // IMPORTANT: This function will be run in a separate worker process! async function setupWorker (id, disconnect) { let exited = false // Wrap stdout and stderr to include the worker ID as a static prefix // console.log('hi') => '[worker.1]: hi' const prefix = `[worker.${id}]: ` prefixStreamWrite(process.stdout, prefix) prefixStreamWrite(process.stderr, prefix) process.on('beforeExit', () => { console.log('Exiting!') }) process.on('SIGTERM', shutdown) process.on('SIGINT', shutdown) console.log('Starting up worker...') // Load the server in each worker process and share the port via sharding const app = require('./lib/app') const warmServer = require('./lib/warm-server') // If in a deployed environment... if (NODE_ENV === 'production') { // If in a true production environment, wait for the cache to be fully warmed. if (process.env.HEROKU_PRODUCTION_APP || process.env.GITHUB_ACTIONS) { await warmServer() } } // Workaround for https://github.com/expressjs/express/issues/1101 const server = require('http').createServer(app) server .listen(port, () => console.log(`app running on http://localhost:${port}`)) .on('error', () => server.close()) function shutdown () { if (exited) return exited = true console.log('Shutting down worker...') disconnect() } } function calculateWorkerCount () { // Heroku's recommended WEB_CONCURRENCY count based on the WEB_MEMORY config const { WEB_CONCURRENCY } = process.env const recommendedCount = parseInt(WEB_CONCURRENCY, 10) || 1 const cpuCount = os.cpus().length // Ensure the recommended count is AT LEAST 1 for safety let workerCount = Math.max(recommendedCount, 1) // Let's do some math... // If in a deployed environment... if (NODE_ENV === 'production') { // If WEB_MEMORY or WEB_CONCURRENCY values were configured in Heroku, use // the smaller value between their recommendation vs. the CPU count if (WEB_CONCURRENCY) { workerCount = Math.min(recommendedCount, cpuCount) } else { workerCount = cpuCount } } return workerCount }