// Copyright Joyent, Inc. and other Node contributors. // // 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. process.env.NODE_DEBUGGER_TIMEOUT = 2000; var common = require('../common'); var assert = require('assert'); var spawn = require('child_process').spawn; var debug = require('_debugger'); var port = common.PORT + 1337; var script = common.fixturesDir + '/breakpoints.js'; var child = spawn(process.execPath, ['debug', '--port=' + port, script], { env: { NODE_FORCE_READLINE: 1 } }); console.error('./node', 'debug', '--port=' + port, script); var buffer = ''; child.stdout.setEncoding('utf-8'); child.stdout.on('data', function(data) { data = (buffer + data.toString()).split(/\n/g); buffer = data.pop(); data.forEach(function(line) { child.emit('line', line); }); }); child.stderr.pipe(process.stdout); var expected = []; child.on('line', function(line) { line = line.replace(/^(debug> )+/, 'debug> '); line = line.replace(/\u001b\[\d+\w/g, ''); console.error('line> ' + line); assert.ok(expected.length > 0, 'Got unexpected line: ' + line); var expectedLine = expected[0].lines.shift(); assert.ok(line.match(expectedLine) !== null, line + ' != ' + expectedLine); if (expected[0].lines.length === 0) { var callback = expected[0].callback; expected.shift(); callback && callback(); } }); function addTest(input, output) { function next() { if (expected.length > 0) { var res = child.stdin.write(expected[0].input + '\n'), callback; if (!expected[0].lines) { callback = expected[0].callback; expected.shift(); } if (callback) { if (res !== true) { child.stdin.on('drain', callback); } else { process.nextTick(callback); } } } else { finish(); } }; expected.push({input: input, lines: output, callback: next}); } // Initial lines addTest(null, [ /listening on port \d+/, /connecting... ok/, /break in .*:1/, /1/, /2/, /3/ ]); // Next addTest('n', [ /debug> n/, /break in .*:11/, /9/, /10/, /11/, /12/, /13/ ]); // Watch addTest('watch("\'x\'"), true', [/debug>/, /true/]); // Continue addTest('c', [ /debug>/, /break in .*:5/, /Watchers/, /0:\s+'x' = "x"/, /()/, /3/, /4/, /5/, /6/, /7/ ]); // Show watchers addTest('watchers', [ /debug>/, /0:\s+'x' = "x"/ ]); // Unwatch addTest('unwatch("\'x\'"), true', [/debug>/, /true/]); // Step out addTest('o', [ /debug>/, /break in .*:12/, /10/, /11/, /12/, /13/, /14/ ]); // Continue addTest('c', [ /debug>/, /break in .*:5/, /3/, /4/, /5/, /6/, /7/ ]); // Set breakpoint by function name addTest('sb("setInterval()", "!(setInterval.flag++)")', [ /debug>/, /1/, /2/, /3/, /4/, /5/, /6/, /7/, /8/, /9/, /10/ ]); // Continue addTest('c', [ /debug>/, /break in node.js:\d+/, /\d/, /\d/, /\d/, /\d/, /\d/ ]); // Repeat last command addTest('', [ /debug>/, /break in .*breakpoints.js:\d+/, /\d/, /\d/, /\d/, /\d/, /\d/ ]); addTest('repl', [ /debug>/, /Press Ctrl \+ C to leave debug repl/ ]); addTest('now', [ /> now/, /\w* \w* \d* \d* \d*:\d*:\d* GMT[+-]\d* (\w*)/ ]); function finish() { // Exit debugger repl child.kill('SIGINT'); child.kill('SIGINT'); // Exit debugger child.kill('SIGINT'); process.exit(0); } function quit() { if (quit.called) return; quit.called = true; child.stdin.write('quit'); } setTimeout(function() { var err = 'Timeout'; if (expected.length > 0 && expected[0].lines) { err = err + '. Expected: ' + expected[0].lines.shift(); } quit(); child.kill('SIGINT'); child.kill('SIGTERM'); // give the sigkill time to work. setTimeout(function() { throw new Error(err); }, 100); }, 5000); process.once('uncaughtException', function(e) { quit(); console.error(e.toString()); process.exit(1); }); process.on('exit', function(code) { quit(); if (code === 0) { assert.equal(expected.length, 0); } });