import child_process from 'child_process'; import { once } from 'events'; import fs from 'fs'; import rimraf from 'rimraf'; import path from 'path'; import os from 'os'; import mkdirp from 'mkdirp'; async function sleep(ms: number) { return new Promise(resolve => setTimeout(resolve, ms)); } const EXIT_FILE = '.exit'; const UPDATE_FILE = '.update'; async function runCommand(command: string, ...args: string[]) { console.log('running', command, ...args); const cp = child_process.spawn(command, args, { stdio: 'inherit' }); await once(cp, 'exit'); } async function runCommandEatError(command: string, ...args: string[]) { try { await runCommand(command, ...args); } catch (e) { console.warn(command, args, 'command exited with error, ignoring', e) } } export async function runServer() { console.log('Starting scrypted main...'); await runCommand('npm', 'exec', 'scrypted-serve'); } async function startServer() { try { await runServer(); } catch (e) { console.error('scrypted server exited with error', e); } } export function getInstallDir() { return process.env.SCRYPTED_INSTALL_PATH || path.join(os.homedir(), '.scrypted'); } export function cwdInstallDir(): { volume: string, installDir: string } { const installDir = getInstallDir(); const volume = path.join(installDir, 'volume'); mkdirp.sync(volume); process.chdir(installDir); return { volume, installDir }; } export async function installServe() { const { installDir } = cwdInstallDir(); // something about low memory environments is causing npm install of node-pty to fail? // maybe too many concurrent post install things happening and node-gyp is failing to build? // not entirely sure i need this anymore, since I fixed various other install related // low memory issues. await runCommandEatError('npm', '--prefix', installDir, 'install', '--production', 'node-pty'); await runCommandEatError('npm', '--prefix', installDir, 'install', '--production', '@scrypted/server@latest'); return installDir; } export async function serveMain(install: boolean) { const { installDir, volume } = cwdInstallDir(); if (!fs.existsSync('node_modules/@scrypted/server')) { install = true; console.log('Package @scrypted/server not found. Installing.'); } if (install) { await installServe(); } process.env.SCRYPTED_NPM_SERVE = 'true'; process.env.SCRYPTED_VOLUME = volume; process.env.SCRYPTED_CAN_EXIT = 'true'; process.env.SCRYPTED_CAN_RESTART = 'true'; console.log('cwd', process.cwd()); while (true) { rimraf.sync(EXIT_FILE); rimraf.sync(UPDATE_FILE); await startServer(); if (fs.existsSync(EXIT_FILE)) { console.log('Exiting.'); process.exit(); } else if (fs.existsSync(UPDATE_FILE)) { console.log('Update requested. Installing.'); await runCommandEatError('npm', '--prefix', installDir, 'install', '--production', '@scrypted/server@latest'); } else { console.log(`Service exited. Restarting momentarily.`); await sleep(10000); } } }