diff --git a/README.md b/README.md index d29dd26..0b5abfc 100644 --- a/README.md +++ b/README.md @@ -34,6 +34,11 @@ try { }); tracer.trace('github.com'); + + // and you can terminate traceroute by calling `kill` method + setTimeout(() => { + tracer.kill() + }, 5000) } catch (ex) { console.log(ex); } diff --git a/src/process.ts b/src/process.ts index 9843c83..85ea4fc 100644 --- a/src/process.ts +++ b/src/process.ts @@ -2,7 +2,8 @@ import events from 'events'; import readline from 'readline'; import validator from 'validator'; -import { spawn } from 'child_process'; +import { ChildProcessWithoutNullStreams, spawn } from 'child_process'; +import { Readable } from 'stream'; export interface Hop { hop: number; @@ -13,10 +14,35 @@ export interface Hop { } export abstract class Process extends events.EventEmitter { + process: ChildProcessWithoutNullStreams|null = null + constructor(private command: string, private args: string[]) { super(); } + private readProcessStd(input: Readable) { + let isDestinationCaptured = false + readline.createInterface({ + input, + terminal: false + }) + .on('line', (line) => { + if (!isDestinationCaptured) { + const destination = this.parseDestination(line); + if (destination !== null) { + this.emit('destination', destination); + + isDestinationCaptured = true; + } + } + + const hop = this.parseHop(line); + if (hop !== null) { + this.emit('hop', hop); + } + }); + } + public trace(domainName: string): void { if (!this.isValidDomainName(domainName)) { throw "Invalid domain name or IP address"; @@ -24,34 +50,24 @@ export abstract class Process extends events.EventEmitter { this.args.push(domainName); - const process = spawn(this.command, this.args); - process.on('close', (code) => { + this.process = spawn(this.command, this.args); + this.process.on('close', (code) => { this.emit('close', code); }); - this.emit('pid', process.pid); - - let isDestinationCaptured = false; - if (process.pid) { - readline.createInterface({ - input: process.stdout, - terminal: false - }) - .on('line', (line) => { - if (!isDestinationCaptured) { - const destination = this.parseDestination(line); - if (destination !== null) { - this.emit('destination', destination); + this.emit('pid', this.process.pid); - isDestinationCaptured = true; - } - } + if (this.process.pid) { + this.readProcessStd(this.process.stdout) + this.readProcessStd(this.process.stderr) + } + } - const hop = this.parseHop(line); - if (hop !== null) { - this.emit('hop', hop); - } - }); + public kill(signal?: NodeJS.Signals | number) { + if (this.process) { + const result = this.process.kill(signal) + this.removeAllListeners() + return result } } diff --git a/test/index.spec.ts b/test/index.spec.ts index f6d5286..fe52751 100644 --- a/test/index.spec.ts +++ b/test/index.spec.ts @@ -3,29 +3,42 @@ import validator from 'validator'; import Traceroute from '../src/index'; describe('Traceroute', () => { - it('should verify pid, destination, hops and close code', (wait) => { - const tracer = new Traceroute(); - - tracer - .on('pid', (pid) => { - expect(Number.isInteger(pid)).toBeTruthy(); - }) - .on('destination', (destination) => { - expect(validator.isIP(destination)).toBeTruthy(); - }) - .on('hop', (hopObj) => { - const { hop, ip, rtt1 } = hopObj; - - expect(Number.isInteger(hop)).toBeTruthy(); - expect(validator.isIP(ip) || ip === '*').toBeTruthy(); - expect(/^\d+\.\d+\sms$/.test(rtt1) || rtt1 === '*').toBeTruthy(); - }) - .on('close', (code) => { - expect(Number.isInteger(code)).toBeTruthy(); - - wait(); - }); - - tracer.trace('github.com'); - }, 60000); -}); \ No newline at end of file + it('should verify pid, destination, hops and close code', (wait) => { + const tracer = new Traceroute(); + + tracer + .on('pid', (pid) => { + expect(Number.isInteger(pid)).toBeTruthy(); + }) + .on('destination', (destination) => { + expect(validator.isIP(destination)).toBeTruthy(); + }) + .on('hop', (hopObj) => { + const { hop, ip, rtt1 } = hopObj; + + expect(Number.isInteger(hop)).toBeTruthy(); + expect(validator.isIP(ip) || ip === '*').toBeTruthy(); + expect(/^\d+\.\d+\sms$/.test(rtt1) || rtt1 === '*').toBeTruthy(); + }) + .on('close', (code) => { + expect(Number.isInteger(code)).toBeTruthy(); + + wait(); + }); + + tracer.trace('github.com'); + }, 60000); + + it('should exit trace by calling kill method', (done) => { + const tracer = new Traceroute(); + + tracer.trace('github.com'); + + setTimeout(() => { + const killResult = tracer.kill() + expect(tracer.process?.killed).toBeTruthy() + expect(killResult).toBeTruthy() + done() + }, 5000) + }, 10000); +});