import * as plugins from './tswatch.plugins.js'; import { logger } from './tswatch.logging.js'; export interface IWatcherConstructorOptions { filePathToWatch: string; commandToExecute?: string; functionToCall?: () => Promise; timeout?: number; } /** * A watcher keeps track of one child execution */ export class Watcher { /** * used to execute shell commands */ private smartshellInstance = new plugins.smartshell.Smartshell({ executor: 'bash', }); private currentExecution: plugins.smartshell.IExecResultStreaming; private smartchokWatcher = new plugins.smartchok.Smartchok([]); private options: IWatcherConstructorOptions; constructor(optionsArg: IWatcherConstructorOptions) { this.options = optionsArg; } /** * start the file */ public async start() { await this.setupCleanup(); console.log(`Looking at ${this.options.filePathToWatch} for changes`); this.smartchokWatcher.add([this.options.filePathToWatch]); // __dirname refers to the directory of this very file await this.smartchokWatcher.start(); const changeObservable = await this.smartchokWatcher.getObservableFor('change'); changeObservable.subscribe(() => { this.updateCurrentExecution(); }); await this.updateCurrentExecution(); logger.log('info', `watcher started for ${this.options.filePathToWatch}`); } /** * updates the current execution */ private async updateCurrentExecution() { if (this.options.commandToExecute) { if (this.currentExecution) { logger.log('ok', `reexecuting ${this.options.commandToExecute}`); this.currentExecution.kill(); } else { logger.log('ok', `executing ${this.options.commandToExecute} for the first time`); } this.currentExecution = await this.smartshellInstance.execStreaming( this.options.commandToExecute ); } else { console.log('no executionCommand set'); } if (this.options.functionToCall) { this.options.functionToCall(); } else { console.log('no functionToCall set.') } } /** * this method sets up a clean exit strategy */ private async setupCleanup() { process.on('exit', () => { console.log(''); console.log('now exiting!'); this.stop(); process.exit(0); }); process.on('SIGINT', () => { console.log(''); console.log('ok! got SIGINT We are exiting! Just cleaning up to exit neatly :)'); this.stop(); process.exit(0); }); // handle timeout if (this.options.timeout) { plugins.smartdelay.delayFor(this.options.timeout).then(() => { console.log(`timed out afer ${this.options.timeout} milliseconds! exiting!`); this.stop(); process.exit(0); }); } } /** * stops the watcher */ public async stop() { await this.smartchokWatcher.stop(); if (this.currentExecution && !this.currentExecution.childProcess.killed) { this.currentExecution.kill(); } } }