tswatch/ts/tswatch.classes.watcher.ts

107 lines
2.9 KiB
TypeScript
Raw Normal View History

2019-05-09 00:08:40 +02:00
import * as plugins from './tswatch.plugins';
import { logger } from './tswatch.logging';
2019-10-12 16:54:03 +02:00
export type TCommandFunction = () => Promise<void>;
2019-05-09 00:08:40 +02:00
export interface IWatcherConstructorOptions {
filePathToWatch: string;
2019-10-12 16:54:03 +02:00
commandToExecute: string | TCommandFunction;
2019-05-09 00:08:40 +02:00
timeout?: number;
}
/**
* A watcher keeps track of one child execution
*/
export class Watcher {
2019-10-12 16:54:03 +02:00
/**
* used to execute shell commands
*/
2019-05-09 00:08:40 +02:00
private smartshellInstance = new plugins.smartshell.Smartshell({
2020-07-04 10:45:56 +00:00
executor: 'bash',
2019-05-09 00:08:40 +02:00
});
2019-10-12 16:54:03 +02:00
2019-05-09 00:08:40 +02:00
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() {
2019-10-12 15:07:44 +02:00
await this.setupCleanup();
2019-05-09 00:08:40 +02:00
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();
});
2019-10-14 14:53:55 +02:00
await this.updateCurrentExecution();
2019-05-09 00:08:40 +02:00
}
/**
* updates the current execution
*/
private async updateCurrentExecution() {
2019-10-12 16:54:03 +02:00
if (typeof this.options.commandToExecute === 'string') {
if (this.currentExecution) {
logger.log('ok', `reexecuting ${this.options.commandToExecute}`);
2020-05-22 07:19:39 +00:00
this.currentExecution.kill();
2019-10-12 16:54:03 +02:00
} else {
logger.log('ok', `executing ${this.options.commandToExecute} for the first time`);
}
this.currentExecution = await this.smartshellInstance.execStreaming(
this.options.commandToExecute
);
2019-05-09 00:08:40 +02:00
} else {
2020-05-22 00:53:11 +00:00
console.log('cannot run execution task');
2019-05-09 00:08:40 +02:00
}
}
/**
* this method sets up a clean exit strategy
*/
private async setupCleanup() {
const cleanup = () => {
if (this.currentExecution) {
process.kill(-this.currentExecution.childProcess.pid);
}
};
process.on('exit', () => {
console.log('');
console.log('now exiting!');
cleanup();
process.exit(0);
});
process.on('SIGINT', () => {
console.log('');
console.log('ok! got SIGINT We are exiting! Just cleaning up to exit neatly :)');
cleanup();
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!`);
cleanup();
process.exit(0);
});
}
}
2019-10-12 15:07:44 +02:00
2019-05-09 00:08:40 +02:00
/**
* stops the watcher
*/
public async stop() {
await this.smartchokWatcher.stop();
2019-05-27 15:38:07 +02:00
if (this.currentExecution && !this.currentExecution.childProcess.killed) {
2019-05-09 00:08:40 +02:00
process.kill(-this.currentExecution.childProcess.pid);
}
}
}