573 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			573 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | "use strict"; | ||
|  | var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { | ||
|  |     return new (P || (P = Promise))(function (resolve, reject) { | ||
|  |         function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } | ||
|  |         function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } | ||
|  |         function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); } | ||
|  |         step((generator = generator.apply(thisArg, _arguments || [])).next()); | ||
|  |     }); | ||
|  | }; | ||
|  | Object.defineProperty(exports, "__esModule", { value: true }); | ||
|  | const os = require("os"); | ||
|  | const events = require("events"); | ||
|  | const child = require("child_process"); | ||
|  | /* eslint-disable @typescript-eslint/unbound-method */ | ||
|  | const IS_WINDOWS = process.platform === 'win32'; | ||
|  | /* | ||
|  |  * Class for running command line tools. Handles quoting and arg parsing in a platform agnostic way. | ||
|  |  */ | ||
|  | class ToolRunner extends events.EventEmitter { | ||
|  |     constructor(toolPath, args, options) { | ||
|  |         super(); | ||
|  |         if (!toolPath) { | ||
|  |             throw new Error("Parameter 'toolPath' cannot be null or empty."); | ||
|  |         } | ||
|  |         this.toolPath = toolPath; | ||
|  |         this.args = args || []; | ||
|  |         this.options = options || {}; | ||
|  |     } | ||
|  |     _debug(message) { | ||
|  |         if (this.options.listeners && this.options.listeners.debug) { | ||
|  |             this.options.listeners.debug(message); | ||
|  |         } | ||
|  |     } | ||
|  |     _getCommandString(options, noPrefix) { | ||
|  |         const toolPath = this._getSpawnFileName(); | ||
|  |         const args = this._getSpawnArgs(options); | ||
|  |         let cmd = noPrefix ? '' : '[command]'; // omit prefix when piped to a second tool
 | ||
|  |         if (IS_WINDOWS) { | ||
|  |             // Windows + cmd file
 | ||
|  |             if (this._isCmdFile()) { | ||
|  |                 cmd += toolPath; | ||
|  |                 for (const a of args) { | ||
|  |                     cmd += ` ${a}`; | ||
|  |                 } | ||
|  |             } | ||
|  |             // Windows + verbatim
 | ||
|  |             else if (options.windowsVerbatimArguments) { | ||
|  |                 cmd += `"${toolPath}"`; | ||
|  |                 for (const a of args) { | ||
|  |                     cmd += ` ${a}`; | ||
|  |                 } | ||
|  |             } | ||
|  |             // Windows (regular)
 | ||
|  |             else { | ||
|  |                 cmd += this._windowsQuoteCmdArg(toolPath); | ||
|  |                 for (const a of args) { | ||
|  |                     cmd += ` ${this._windowsQuoteCmdArg(a)}`; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         else { | ||
|  |             // OSX/Linux - this can likely be improved with some form of quoting.
 | ||
|  |             // creating processes on Unix is fundamentally different than Windows.
 | ||
|  |             // on Unix, execvp() takes an arg array.
 | ||
|  |             cmd += toolPath; | ||
|  |             for (const a of args) { | ||
|  |                 cmd += ` ${a}`; | ||
|  |             } | ||
|  |         } | ||
|  |         return cmd; | ||
|  |     } | ||
|  |     _processLineBuffer(data, strBuffer, onLine) { | ||
|  |         try { | ||
|  |             let s = strBuffer + data.toString(); | ||
|  |             let n = s.indexOf(os.EOL); | ||
|  |             while (n > -1) { | ||
|  |                 const line = s.substring(0, n); | ||
|  |                 onLine(line); | ||
|  |                 // the rest of the string ...
 | ||
|  |                 s = s.substring(n + os.EOL.length); | ||
|  |                 n = s.indexOf(os.EOL); | ||
|  |             } | ||
|  |             strBuffer = s; | ||
|  |         } | ||
|  |         catch (err) { | ||
|  |             // streaming lines to console is best effort.  Don't fail a build.
 | ||
|  |             this._debug(`error processing line. Failed with error ${err}`); | ||
|  |         } | ||
|  |     } | ||
|  |     _getSpawnFileName() { | ||
|  |         if (IS_WINDOWS) { | ||
|  |             if (this._isCmdFile()) { | ||
|  |                 return process.env['COMSPEC'] || 'cmd.exe'; | ||
|  |             } | ||
|  |         } | ||
|  |         return this.toolPath; | ||
|  |     } | ||
|  |     _getSpawnArgs(options) { | ||
|  |         if (IS_WINDOWS) { | ||
|  |             if (this._isCmdFile()) { | ||
|  |                 let argline = `/D /S /C "${this._windowsQuoteCmdArg(this.toolPath)}`; | ||
|  |                 for (const a of this.args) { | ||
|  |                     argline += ' '; | ||
|  |                     argline += options.windowsVerbatimArguments | ||
|  |                         ? a | ||
|  |                         : this._windowsQuoteCmdArg(a); | ||
|  |                 } | ||
|  |                 argline += '"'; | ||
|  |                 return [argline]; | ||
|  |             } | ||
|  |         } | ||
|  |         return this.args; | ||
|  |     } | ||
|  |     _endsWith(str, end) { | ||
|  |         return str.endsWith(end); | ||
|  |     } | ||
|  |     _isCmdFile() { | ||
|  |         const upperToolPath = this.toolPath.toUpperCase(); | ||
|  |         return (this._endsWith(upperToolPath, '.CMD') || | ||
|  |             this._endsWith(upperToolPath, '.BAT')); | ||
|  |     } | ||
|  |     _windowsQuoteCmdArg(arg) { | ||
|  |         // for .exe, apply the normal quoting rules that libuv applies
 | ||
|  |         if (!this._isCmdFile()) { | ||
|  |             return this._uvQuoteCmdArg(arg); | ||
|  |         } | ||
|  |         // otherwise apply quoting rules specific to the cmd.exe command line parser.
 | ||
|  |         // the libuv rules are generic and are not designed specifically for cmd.exe
 | ||
|  |         // command line parser.
 | ||
|  |         //
 | ||
|  |         // for a detailed description of the cmd.exe command line parser, refer to
 | ||
|  |         // http://stackoverflow.com/questions/4094699/how-does-the-windows-command-interpreter-cmd-exe-parse-scripts/7970912#7970912
 | ||
|  |         // need quotes for empty arg
 | ||
|  |         if (!arg) { | ||
|  |             return '""'; | ||
|  |         } | ||
|  |         // determine whether the arg needs to be quoted
 | ||
|  |         const cmdSpecialChars = [ | ||
|  |             ' ', | ||
|  |             '\t', | ||
|  |             '&', | ||
|  |             '(', | ||
|  |             ')', | ||
|  |             '[', | ||
|  |             ']', | ||
|  |             '{', | ||
|  |             '}', | ||
|  |             '^', | ||
|  |             '=', | ||
|  |             ';', | ||
|  |             '!', | ||
|  |             "'", | ||
|  |             '+', | ||
|  |             ',', | ||
|  |             '`', | ||
|  |             '~', | ||
|  |             '|', | ||
|  |             '<', | ||
|  |             '>', | ||
|  |             '"' | ||
|  |         ]; | ||
|  |         let needsQuotes = false; | ||
|  |         for (const char of arg) { | ||
|  |             if (cmdSpecialChars.some(x => x === char)) { | ||
|  |                 needsQuotes = true; | ||
|  |                 break; | ||
|  |             } | ||
|  |         } | ||
|  |         // short-circuit if quotes not needed
 | ||
|  |         if (!needsQuotes) { | ||
|  |             return arg; | ||
|  |         } | ||
|  |         // the following quoting rules are very similar to the rules that by libuv applies.
 | ||
|  |         //
 | ||
|  |         // 1) wrap the string in quotes
 | ||
|  |         //
 | ||
|  |         // 2) double-up quotes - i.e. " => ""
 | ||
|  |         //
 | ||
|  |         //    this is different from the libuv quoting rules. libuv replaces " with \", which unfortunately
 | ||
|  |         //    doesn't work well with a cmd.exe command line.
 | ||
|  |         //
 | ||
|  |         //    note, replacing " with "" also works well if the arg is passed to a downstream .NET console app.
 | ||
|  |         //    for example, the command line:
 | ||
|  |         //          foo.exe "myarg:""my val"""
 | ||
|  |         //    is parsed by a .NET console app into an arg array:
 | ||
|  |         //          [ "myarg:\"my val\"" ]
 | ||
|  |         //    which is the same end result when applying libuv quoting rules. although the actual
 | ||
|  |         //    command line from libuv quoting rules would look like:
 | ||
|  |         //          foo.exe "myarg:\"my val\""
 | ||
|  |         //
 | ||
|  |         // 3) double-up slashes that preceed a quote,
 | ||
|  |         //    e.g.  hello \world    => "hello \world"
 | ||
|  |         //          hello\"world    => "hello\\""world"
 | ||
|  |         //          hello\\"world   => "hello\\\\""world"
 | ||
|  |         //          hello world\    => "hello world\\"
 | ||
|  |         //
 | ||
|  |         //    technically this is not required for a cmd.exe command line, or the batch argument parser.
 | ||
|  |         //    the reasons for including this as a .cmd quoting rule are:
 | ||
|  |         //
 | ||
|  |         //    a) this is optimized for the scenario where the argument is passed from the .cmd file to an
 | ||
|  |         //       external program. many programs (e.g. .NET console apps) rely on the slash-doubling rule.
 | ||
|  |         //
 | ||
|  |         //    b) it's what we've been doing previously (by deferring to node default behavior) and we
 | ||
|  |         //       haven't heard any complaints about that aspect.
 | ||
|  |         //
 | ||
|  |         // note, a weakness of the quoting rules chosen here, is that % is not escaped. in fact, % cannot be
 | ||
|  |         // escaped when used on the command line directly - even though within a .cmd file % can be escaped
 | ||
|  |         // by using %%.
 | ||
|  |         //
 | ||
|  |         // the saving grace is, on the command line, %var% is left as-is if var is not defined. this contrasts
 | ||
|  |         // the line parsing rules within a .cmd file, where if var is not defined it is replaced with nothing.
 | ||
|  |         //
 | ||
|  |         // one option that was explored was replacing % with ^% - i.e. %var% => ^%var^%. this hack would
 | ||
|  |         // often work, since it is unlikely that var^ would exist, and the ^ character is removed when the
 | ||
|  |         // variable is used. the problem, however, is that ^ is not removed when %* is used to pass the args
 | ||
|  |         // to an external program.
 | ||
|  |         //
 | ||
|  |         // an unexplored potential solution for the % escaping problem, is to create a wrapper .cmd file.
 | ||
|  |         // % can be escaped within a .cmd file.
 | ||
|  |         let reverse = '"'; | ||
|  |         let quoteHit = true; | ||
|  |         for (let i = arg.length; i > 0; i--) { | ||
|  |             // walk the string in reverse
 | ||
|  |             reverse += arg[i - 1]; | ||
|  |             if (quoteHit && arg[i - 1] === '\\') { | ||
|  |                 reverse += '\\'; // double the slash
 | ||
|  |             } | ||
|  |             else if (arg[i - 1] === '"') { | ||
|  |                 quoteHit = true; | ||
|  |                 reverse += '"'; // double the quote
 | ||
|  |             } | ||
|  |             else { | ||
|  |                 quoteHit = false; | ||
|  |             } | ||
|  |         } | ||
|  |         reverse += '"'; | ||
|  |         return reverse | ||
|  |             .split('') | ||
|  |             .reverse() | ||
|  |             .join(''); | ||
|  |     } | ||
|  |     _uvQuoteCmdArg(arg) { | ||
|  |         // Tool runner wraps child_process.spawn() and needs to apply the same quoting as
 | ||
|  |         // Node in certain cases where the undocumented spawn option windowsVerbatimArguments
 | ||
|  |         // is used.
 | ||
|  |         //
 | ||
|  |         // Since this function is a port of quote_cmd_arg from Node 4.x (technically, lib UV,
 | ||
|  |         // see https://github.com/nodejs/node/blob/v4.x/deps/uv/src/win/process.c for details),
 | ||
|  |         // pasting copyright notice from Node within this function:
 | ||
|  |         //
 | ||
|  |         //      Copyright Joyent, Inc. and other Node contributors. All rights reserved.
 | ||
|  |         //
 | ||
|  |         //      Permission is hereby granted, free of charge, to any person obtaining a copy
 | ||
|  |         //      of this software and associated documentation files (the "Software"), to
 | ||
|  |         //      deal in the Software without restriction, including without limitation the
 | ||
|  |         //      rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
 | ||
|  |         //      sell copies of the Software, and to permit persons to whom the Software is
 | ||
|  |         //      furnished to do so, subject to the following conditions:
 | ||
|  |         //
 | ||
|  |         //      The above copyright notice and this permission notice shall be included in
 | ||
|  |         //      all copies or substantial portions of the Software.
 | ||
|  |         //
 | ||
|  |         //      THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | ||
|  |         //      IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | ||
|  |         //      FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | ||
|  |         //      AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | ||
|  |         //      LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 | ||
|  |         //      FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
 | ||
|  |         //      IN THE SOFTWARE.
 | ||
|  |         if (!arg) { | ||
|  |             // Need double quotation for empty argument
 | ||
|  |             return '""'; | ||
|  |         } | ||
|  |         if (!arg.includes(' ') && !arg.includes('\t') && !arg.includes('"')) { | ||
|  |             // No quotation needed
 | ||
|  |             return arg; | ||
|  |         } | ||
|  |         if (!arg.includes('"') && !arg.includes('\\')) { | ||
|  |             // No embedded double quotes or backslashes, so I can just wrap
 | ||
|  |             // quote marks around the whole thing.
 | ||
|  |             return `"${arg}"`; | ||
|  |         } | ||
|  |         // Expected input/output:
 | ||
|  |         //   input : hello"world
 | ||
|  |         //   output: "hello\"world"
 | ||
|  |         //   input : hello""world
 | ||
|  |         //   output: "hello\"\"world"
 | ||
|  |         //   input : hello\world
 | ||
|  |         //   output: hello\world
 | ||
|  |         //   input : hello\\world
 | ||
|  |         //   output: hello\\world
 | ||
|  |         //   input : hello\"world
 | ||
|  |         //   output: "hello\\\"world"
 | ||
|  |         //   input : hello\\"world
 | ||
|  |         //   output: "hello\\\\\"world"
 | ||
|  |         //   input : hello world\
 | ||
|  |         //   output: "hello world\\" - note the comment in libuv actually reads "hello world\"
 | ||
|  |         //                             but it appears the comment is wrong, it should be "hello world\\"
 | ||
|  |         let reverse = '"'; | ||
|  |         let quoteHit = true; | ||
|  |         for (let i = arg.length; i > 0; i--) { | ||
|  |             // walk the string in reverse
 | ||
|  |             reverse += arg[i - 1]; | ||
|  |             if (quoteHit && arg[i - 1] === '\\') { | ||
|  |                 reverse += '\\'; | ||
|  |             } | ||
|  |             else if (arg[i - 1] === '"') { | ||
|  |                 quoteHit = true; | ||
|  |                 reverse += '\\'; | ||
|  |             } | ||
|  |             else { | ||
|  |                 quoteHit = false; | ||
|  |             } | ||
|  |         } | ||
|  |         reverse += '"'; | ||
|  |         return reverse | ||
|  |             .split('') | ||
|  |             .reverse() | ||
|  |             .join(''); | ||
|  |     } | ||
|  |     _cloneExecOptions(options) { | ||
|  |         options = options || {}; | ||
|  |         const result = { | ||
|  |             cwd: options.cwd || process.cwd(), | ||
|  |             env: options.env || process.env, | ||
|  |             silent: options.silent || false, | ||
|  |             windowsVerbatimArguments: options.windowsVerbatimArguments || false, | ||
|  |             failOnStdErr: options.failOnStdErr || false, | ||
|  |             ignoreReturnCode: options.ignoreReturnCode || false, | ||
|  |             delay: options.delay || 10000 | ||
|  |         }; | ||
|  |         result.outStream = options.outStream || process.stdout; | ||
|  |         result.errStream = options.errStream || process.stderr; | ||
|  |         return result; | ||
|  |     } | ||
|  |     _getSpawnOptions(options, toolPath) { | ||
|  |         options = options || {}; | ||
|  |         const result = {}; | ||
|  |         result.cwd = options.cwd; | ||
|  |         result.env = options.env; | ||
|  |         result['windowsVerbatimArguments'] = | ||
|  |             options.windowsVerbatimArguments || this._isCmdFile(); | ||
|  |         if (options.windowsVerbatimArguments) { | ||
|  |             result.argv0 = `"${toolPath}"`; | ||
|  |         } | ||
|  |         return result; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Exec a tool. | ||
|  |      * Output will be streamed to the live console. | ||
|  |      * Returns promise with return code | ||
|  |      * | ||
|  |      * @param     tool     path to tool to exec | ||
|  |      * @param     options  optional exec options.  See ExecOptions | ||
|  |      * @returns   number | ||
|  |      */ | ||
|  |     exec() { | ||
|  |         return __awaiter(this, void 0, void 0, function* () { | ||
|  |             return new Promise((resolve, reject) => { | ||
|  |                 this._debug(`exec tool: ${this.toolPath}`); | ||
|  |                 this._debug('arguments:'); | ||
|  |                 for (const arg of this.args) { | ||
|  |                     this._debug(`   ${arg}`); | ||
|  |                 } | ||
|  |                 const optionsNonNull = this._cloneExecOptions(this.options); | ||
|  |                 if (!optionsNonNull.silent && optionsNonNull.outStream) { | ||
|  |                     optionsNonNull.outStream.write(this._getCommandString(optionsNonNull) + os.EOL); | ||
|  |                 } | ||
|  |                 const state = new ExecState(optionsNonNull, this.toolPath); | ||
|  |                 state.on('debug', (message) => { | ||
|  |                     this._debug(message); | ||
|  |                 }); | ||
|  |                 const fileName = this._getSpawnFileName(); | ||
|  |                 const cp = child.spawn(fileName, this._getSpawnArgs(optionsNonNull), this._getSpawnOptions(this.options, fileName)); | ||
|  |                 const stdbuffer = ''; | ||
|  |                 if (cp.stdout) { | ||
|  |                     cp.stdout.on('data', (data) => { | ||
|  |                         if (this.options.listeners && this.options.listeners.stdout) { | ||
|  |                             this.options.listeners.stdout(data); | ||
|  |                         } | ||
|  |                         if (!optionsNonNull.silent && optionsNonNull.outStream) { | ||
|  |                             optionsNonNull.outStream.write(data); | ||
|  |                         } | ||
|  |                         this._processLineBuffer(data, stdbuffer, (line) => { | ||
|  |                             if (this.options.listeners && this.options.listeners.stdline) { | ||
|  |                                 this.options.listeners.stdline(line); | ||
|  |                             } | ||
|  |                         }); | ||
|  |                     }); | ||
|  |                 } | ||
|  |                 const errbuffer = ''; | ||
|  |                 if (cp.stderr) { | ||
|  |                     cp.stderr.on('data', (data) => { | ||
|  |                         state.processStderr = true; | ||
|  |                         if (this.options.listeners && this.options.listeners.stderr) { | ||
|  |                             this.options.listeners.stderr(data); | ||
|  |                         } | ||
|  |                         if (!optionsNonNull.silent && | ||
|  |                             optionsNonNull.errStream && | ||
|  |                             optionsNonNull.outStream) { | ||
|  |                             const s = optionsNonNull.failOnStdErr | ||
|  |                                 ? optionsNonNull.errStream | ||
|  |                                 : optionsNonNull.outStream; | ||
|  |                             s.write(data); | ||
|  |                         } | ||
|  |                         this._processLineBuffer(data, errbuffer, (line) => { | ||
|  |                             if (this.options.listeners && this.options.listeners.errline) { | ||
|  |                                 this.options.listeners.errline(line); | ||
|  |                             } | ||
|  |                         }); | ||
|  |                     }); | ||
|  |                 } | ||
|  |                 cp.on('error', (err) => { | ||
|  |                     state.processError = err.message; | ||
|  |                     state.processExited = true; | ||
|  |                     state.processClosed = true; | ||
|  |                     state.CheckComplete(); | ||
|  |                 }); | ||
|  |                 cp.on('exit', (code) => { | ||
|  |                     state.processExitCode = code; | ||
|  |                     state.processExited = true; | ||
|  |                     this._debug(`Exit code ${code} received from tool '${this.toolPath}'`); | ||
|  |                     state.CheckComplete(); | ||
|  |                 }); | ||
|  |                 cp.on('close', (code) => { | ||
|  |                     state.processExitCode = code; | ||
|  |                     state.processExited = true; | ||
|  |                     state.processClosed = true; | ||
|  |                     this._debug(`STDIO streams have closed for tool '${this.toolPath}'`); | ||
|  |                     state.CheckComplete(); | ||
|  |                 }); | ||
|  |                 state.on('done', (error, exitCode) => { | ||
|  |                     if (stdbuffer.length > 0) { | ||
|  |                         this.emit('stdline', stdbuffer); | ||
|  |                     } | ||
|  |                     if (errbuffer.length > 0) { | ||
|  |                         this.emit('errline', errbuffer); | ||
|  |                     } | ||
|  |                     cp.removeAllListeners(); | ||
|  |                     if (error) { | ||
|  |                         reject(error); | ||
|  |                     } | ||
|  |                     else { | ||
|  |                         resolve(exitCode); | ||
|  |                     } | ||
|  |                 }); | ||
|  |             }); | ||
|  |         }); | ||
|  |     } | ||
|  | } | ||
|  | exports.ToolRunner = ToolRunner; | ||
|  | /** | ||
|  |  * Convert an arg string to an array of args. Handles escaping | ||
|  |  * | ||
|  |  * @param    argString   string of arguments | ||
|  |  * @returns  string[]    array of arguments | ||
|  |  */ | ||
|  | function argStringToArray(argString) { | ||
|  |     const args = []; | ||
|  |     let inQuotes = false; | ||
|  |     let escaped = false; | ||
|  |     let arg = ''; | ||
|  |     function append(c) { | ||
|  |         // we only escape double quotes.
 | ||
|  |         if (escaped && c !== '"') { | ||
|  |             arg += '\\'; | ||
|  |         } | ||
|  |         arg += c; | ||
|  |         escaped = false; | ||
|  |     } | ||
|  |     for (let i = 0; i < argString.length; i++) { | ||
|  |         const c = argString.charAt(i); | ||
|  |         if (c === '"') { | ||
|  |             if (!escaped) { | ||
|  |                 inQuotes = !inQuotes; | ||
|  |             } | ||
|  |             else { | ||
|  |                 append(c); | ||
|  |             } | ||
|  |             continue; | ||
|  |         } | ||
|  |         if (c === '\\' && escaped) { | ||
|  |             append(c); | ||
|  |             continue; | ||
|  |         } | ||
|  |         if (c === '\\' && inQuotes) { | ||
|  |             escaped = true; | ||
|  |             continue; | ||
|  |         } | ||
|  |         if (c === ' ' && !inQuotes) { | ||
|  |             if (arg.length > 0) { | ||
|  |                 args.push(arg); | ||
|  |                 arg = ''; | ||
|  |             } | ||
|  |             continue; | ||
|  |         } | ||
|  |         append(c); | ||
|  |     } | ||
|  |     if (arg.length > 0) { | ||
|  |         args.push(arg.trim()); | ||
|  |     } | ||
|  |     return args; | ||
|  | } | ||
|  | exports.argStringToArray = argStringToArray; | ||
|  | class ExecState extends events.EventEmitter { | ||
|  |     constructor(options, toolPath) { | ||
|  |         super(); | ||
|  |         this.processClosed = false; // tracks whether the process has exited and stdio is closed
 | ||
|  |         this.processError = ''; | ||
|  |         this.processExitCode = 0; | ||
|  |         this.processExited = false; // tracks whether the process has exited
 | ||
|  |         this.processStderr = false; // tracks whether stderr was written to
 | ||
|  |         this.delay = 10000; // 10 seconds
 | ||
|  |         this.done = false; | ||
|  |         this.timeout = null; | ||
|  |         if (!toolPath) { | ||
|  |             throw new Error('toolPath must not be empty'); | ||
|  |         } | ||
|  |         this.options = options; | ||
|  |         this.toolPath = toolPath; | ||
|  |         if (options.delay) { | ||
|  |             this.delay = options.delay; | ||
|  |         } | ||
|  |     } | ||
|  |     CheckComplete() { | ||
|  |         if (this.done) { | ||
|  |             return; | ||
|  |         } | ||
|  |         if (this.processClosed) { | ||
|  |             this._setResult(); | ||
|  |         } | ||
|  |         else if (this.processExited) { | ||
|  |             this.timeout = setTimeout(ExecState.HandleTimeout, this.delay, this); | ||
|  |         } | ||
|  |     } | ||
|  |     _debug(message) { | ||
|  |         this.emit('debug', message); | ||
|  |     } | ||
|  |     _setResult() { | ||
|  |         // determine whether there is an error
 | ||
|  |         let error; | ||
|  |         if (this.processExited) { | ||
|  |             if (this.processError) { | ||
|  |                 error = new Error(`There was an error when attempting to execute the process '${this.toolPath}'. This may indicate the process failed to start. Error: ${this.processError}`); | ||
|  |             } | ||
|  |             else if (this.processExitCode !== 0 && !this.options.ignoreReturnCode) { | ||
|  |                 error = new Error(`The process '${this.toolPath}' failed with exit code ${this.processExitCode}`); | ||
|  |             } | ||
|  |             else if (this.processStderr && this.options.failOnStdErr) { | ||
|  |                 error = new Error(`The process '${this.toolPath}' failed because one or more lines were written to the STDERR stream`); | ||
|  |             } | ||
|  |         } | ||
|  |         // clear the timeout
 | ||
|  |         if (this.timeout) { | ||
|  |             clearTimeout(this.timeout); | ||
|  |             this.timeout = null; | ||
|  |         } | ||
|  |         this.done = true; | ||
|  |         this.emit('done', error, this.processExitCode); | ||
|  |     } | ||
|  |     static HandleTimeout(state) { | ||
|  |         if (state.done) { | ||
|  |             return; | ||
|  |         } | ||
|  |         if (!state.processClosed && state.processExited) { | ||
|  |             const message = `The STDIO streams did not close within ${state.delay / | ||
|  |                 1000} seconds of the exit event from process '${state.toolPath}'. This may indicate a child process inherited the STDIO streams and has not yet exited.`;
 | ||
|  |             state._debug(message); | ||
|  |         } | ||
|  |         state._setResult(); | ||
|  |     } | ||
|  | } | ||
|  | //# sourceMappingURL=toolrunner.js.map
 |