From 25e59e60538df1fb02b464f19297c4723b136d0e Mon Sep 17 00:00:00 2001 From: Yaroslav Serhieiev Date: Thu, 3 May 2018 19:18:48 +0300 Subject: [PATCH] Emulator wrapper fixes (#702) * fix: emulator should always get non-empty command line arguments fix: detox fails to start when another emulator is running * code: changed '-gpu host' to '-gpu auto' out of compatibility concerns * code: removed double error logging, extra join and split, cleaner method chaining --- detox/src/devices/android/Emulator.js | 62 ++++++++++++++++----------- 1 file changed, 36 insertions(+), 26 deletions(-) diff --git a/detox/src/devices/android/Emulator.js b/detox/src/devices/android/Emulator.js index 7e56b1e109..df50e6d898 100644 --- a/detox/src/devices/android/Emulator.js +++ b/detox/src/devices/android/Emulator.js @@ -24,45 +24,55 @@ class Emulator { } async boot(emulatorName) { - const headless = argparse.getArgValue('headless') ? '-no-window' : ''; - const cmd = `-verbose -gpu host -no-audio ${headless} @${emulatorName}`; - log.verbose(this.emulatorBin, cmd); + const emulatorArgs = _.compact([ + '-verbose', + '-gpu', 'auto', + '-no-audio', + argparse.getArgValue('headless') ? '-no-window' : '', + `@${emulatorName}` + ]); + + let childProcessOutput; const tempLog = `./${emulatorName}.log`; const stdout = fs.openSync(tempLog, 'a'); const stderr = fs.openSync(tempLog, 'a'); - const tail = new Tail(tempLog); - const promise = spawn(this.emulatorBin, _.split(cmd, ' '), {detached: true, stdio: ['ignore', stdout, stderr]}); - - const childProcess = promise.childProcess; - childProcess.unref(); - - tail.on("line", function(data) { - if (data.includes('Adb connected, start proxing data')) { - detach(); - } - if (data.includes(`There's another emulator instance running with the current AVD`)) { - detach(); + const tail = new Tail(tempLog).on("line", (line) => { + if (line.includes('Adb connected, start proxing data')) { + childProcessPromise._cpResolve(); } }); - tail.on("error", function(error) { - detach(); - log.verbose('Emulator stderr: ', error); - }); + function detach() { + if (childProcessOutput) { + return; + } - promise.catch(function(err) { - log.error('Emulator ERROR: ', err); - }); + childProcessOutput = fs.readFileSync(tempLog, 'utf8'); - function detach() { tail.unwatch(); fs.closeSync(stdout); fs.closeSync(stderr); - fs.unlink(tempLog, () => {}); - promise._cpResolve(); + fs.unlink(tempLog, _.noop); } - return promise; + log.verbose(this.emulatorBin, ...emulatorArgs); + const childProcessPromise = spawn(this.emulatorBin, emulatorArgs, { detached: true, stdio: ['ignore', stdout, stderr] }); + childProcessPromise.childProcess.unref(); + + return childProcessPromise.catch((err) => { + detach(); + + if (childProcessOutput.includes(`There's another emulator instance running with the current AVD`)) { + return; + } + + log.error('ChildProcessError', '%s', err.message); + log.error('stderr', '%s', childProcessOutput); + throw err; + }).then(() => { + detach(); + log.verbose('stdout', '%s', childProcessOutput); + }); } }