Skip to content
This repository has been archived by the owner on Jul 29, 2024. It is now read-only.

Commit

Permalink
Refactor element explorer to work with selenium-webdriver 3 (#3828)
Browse files Browse the repository at this point in the history
This implementation now relies mostly on promises explicitly,
so the control flow is only used to add one large task to the queue.
This should pave the way for the eventual removal of the control flow,
as well as getting element explorer to work immediately.

BREAKING CHANGE

You can no longer use the `repl` command from within `browser.pause()`. Instead,
use `broser.explore()` to directly enter the repl.
  • Loading branch information
juliemr authored Dec 28, 2016
1 parent 3644f01 commit 530ca59
Show file tree
Hide file tree
Showing 10 changed files with 198 additions and 153 deletions.
10 changes: 10 additions & 0 deletions lib/breakpointhook.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
module.exports = function() {
return true;
};

/**
* The reason this file exists is so that we can set a breakpoint via
* script name, and then control when that breakpoint is set in
* our library code by importing and calling this function. The
* breakpoint will always be on line 2.
*/
43 changes: 25 additions & 18 deletions lib/browser.ts
Original file line number Diff line number Diff line change
Expand Up @@ -972,37 +972,46 @@ export class ProtractorBrowser extends AbstractExtendedWebDriver {
}

/**
* Beta (unstable) enterRepl function for entering the repl loop from
* any point in the control flow. Use browser.enterRepl() in your test.
* See browser.explore().
*/
enterRepl(opt_debugPort?: number) {
return this.explore(opt_debugPort);
}

/**
* Beta (unstable) explore function for entering the repl loop from
* any point in the control flow. Use browser.explore() in your test.
* Does not require changes to the command line (no need to add 'debug').
* Note, if you are wrapping your own instance of Protractor, you must
* expose globals 'browser' and 'protractor' for pause to work.
*
* @example
* element(by.id('foo')).click();
* browser.enterRepl();
* browser.explore();
* // Execution will stop before the next click action.
* element(by.id('bar')).click();
*
* @param {number=} opt_debugPort Optional port to use for the debugging
* process
*/
enterRepl(opt_debugPort?: number) {
explore(opt_debugPort?: number) {
let debuggerClientPath = __dirname + '/debugger/clients/explorer.js';
let onStartFn = () => {
logger.info();
logger.info('------- Element Explorer -------');
logger.info(
'Starting WebDriver debugger in a child process. Element ' +
'Explorer is still beta, please report issues at ' +
'github.com/angular/protractor');
logger.info();
logger.info('Type <tab> to see a list of locator strategies.');
logger.info('Use the `list` helper function to find elements by strategy:');
logger.info(' e.g., list(by.binding(\'\')) gets all bindings.');
let onStartFn = (firstTime: boolean) => {
logger.info();
if (firstTime) {
logger.info('------- Element Explorer -------');
logger.info(
'Starting WebDriver debugger in a child process. Element ' +
'Explorer is still beta, please report issues at ' +
'github.com/angular/protractor');
logger.info();
logger.info('Type <tab> to see a list of locator strategies.');
logger.info('Use the `list` helper function to find elements by strategy:');
logger.info(' e.g., list(by.binding(\'\')) gets all bindings.');
logger.info();
}
};
this.debugHelper.init(debuggerClientPath, onStartFn, opt_debugPort);
this.debugHelper.initBlocking(debuggerClientPath, onStartFn, opt_debugPort);
}

/**
Expand Down Expand Up @@ -1040,8 +1049,6 @@ export class ProtractorBrowser extends AbstractExtendedWebDriver {
logger.info();
logger.info('press c to continue to the next webdriver command');
logger.info('press ^D to detach debugger and resume code execution');
logger.info('type "repl" to enter interactive mode');
logger.info('type "exit" to break out of interactive mode');
logger.info();
}
};
Expand Down
99 changes: 54 additions & 45 deletions lib/debugger.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {Locator} from './locators';
import {Logger} from './logger';
import {Ptor} from './ptor';
import * as helper from './util';
let breakpointHook = require('./breakpointhook.js');

declare var global: any;
declare var process: any;
Expand All @@ -25,32 +26,36 @@ export class DebugHelper {

constructor(private browserUnderDebug_: ProtractorBrowser) {}


initBlocking(debuggerClientPath: string, onStartFn: Function, opt_debugPort?: number) {
this.init_(debuggerClientPath, true, onStartFn, opt_debugPort);
}

init(debuggerClientPath: string, onStartFn: Function, opt_debugPort?: number) {
this.init_(debuggerClientPath, false, onStartFn, opt_debugPort);
}

/**
* 1) Set up helper functions for debugger clients to call on (e.g.
* getControlFlowText, execute code, get autocompletion).
* execute code, get autocompletion).
* 2) Enter process into debugger mode. (i.e. process._debugProcess).
* 3) Invoke the debugger client specified by debuggerClientPath.
*
* @param {string} debuggerClientPath Absolute path of debugger client to use.
* @param {boolean} blockUntilExit Whether to block the flow until process exit or resume
* immediately.
* @param {Function} onStartFn Function to call when the debugger starts. The
* function takes a single parameter, which represents whether this is the
* first time that the debugger is called.
* @param {number=} opt_debugPort Optional port to use for the debugging
* process.
*
* @return {Promise} If blockUntilExit, a promise resolved when the debugger process
* exits. Otherwise, resolved when the debugger process is ready to begin.
*/
init(debuggerClientPath: string, onStartFn: Function, opt_debugPort?: number) {
(wdpromise.ControlFlow as any).prototype.getControlFlowText = function() {
let controlFlowText = this.getSchedule(/* opt_includeStackTraces */ true);
// This filters the entire control flow text, not just the stack trace, so
// unless we maintain a good (i.e. non-generic) set of keywords in
// STACK_SUBSTRINGS_TO_FILTER, we run the risk of filtering out non stack
// trace. The alternative though, which is to reimplement
// webdriver.promise.ControlFlow.prototype.getSchedule() here is much
// hackier, and involves messing with the control flow's internals /
// private variables.
return helper.filterStackTrace(controlFlowText);
};

init_(
debuggerClientPath: string, blockUntilExit: boolean, onStartFn: Function,
opt_debugPort?: number) {
const vm_ = require('vm');
let flow = wdpromise.controlFlow();

Expand All @@ -75,8 +80,11 @@ export class DebugHelper {
}
let sandbox = vm_.createContext(context);

let debuggerReadyPromise = wdpromise.defer();
flow.execute(() => {
let debuggingDone = wdpromise.defer();

// We run one flow.execute block for the debugging session. All
// subcommands should be scheduled under this task.
let executePromise = flow.execute(() => {
process['debugPort'] = opt_debugPort || process['debugPort'];
this.validatePortAvailability_(process['debugPort']).then((firstTime: boolean) => {
onStartFn(firstTime);
Expand All @@ -93,34 +101,30 @@ export class DebugHelper {
.on('message',
(m: string) => {
if (m === 'ready') {
debuggerReadyPromise.fulfill();
breakpointHook();
if (!blockUntilExit) {
debuggingDone.fulfill();
}
}
})
.on('exit', () => {
logger.info('Debugger exiting');
// Clear this so that we know it's ok to attach a debugger
// again.
this.dbgCodeExecutor = null;
debuggingDone.fulfill();
});
});
});

let pausePromise = flow.execute(() => {
return debuggerReadyPromise.promise.then(() => {
// Necessary for backward compatibility with node < 0.12.0
return this.browserUnderDebug_.executeScriptWithDescription('', 'empty debugger hook');
});
});
return debuggingDone.promise;
}, 'debugging tasks');

// Helper used only by debuggers at './debugger/modes/*.js' to insert code
// into the control flow.
// In order to achieve this, we maintain a promise at the top of the control
// into the control flow, via debugger 'evaluate' protocol.
// In order to achieve this, we maintain a task at the top of the control
// flow, so that we can insert frames into it.
// To be able to simulate callback/asynchronous code, we poll this object
// for a result at every run of DeferredExecutor.execute.
let browserUnderDebug = this.browserUnderDebug_;
// whenever `breakpointHook` is called.
this.dbgCodeExecutor = {
execPromise_: pausePromise, // Promise pointing to current stage of flow.
execPromise_: undefined, // Promise pointing to currently executing command.
execPromiseResult_: undefined, // Return value of promise.
execPromiseError_: undefined, // Error from promise.

Expand All @@ -137,20 +141,19 @@ export class DebugHelper {
execute_: function(execFn_: Function) {
this.execPromiseResult_ = this.execPromiseError_ = undefined;

this.execPromise_ = this.execPromise_.then(execFn_).then(
this.execPromise_ = execFn_();
// Note: This needs to be added after setting execPromise to execFn,
// or else we cause this.execPromise_ to get stuck in pending mode
// at our next breakpoint.
this.execPromise_.then(
(result: Object) => {
this.execPromiseResult_ = result;
breakpointHook();
},
(err: Error) => {
this.execPromiseError_ = err;
breakpointHook();
});

// This dummy command is necessary so that the DeferredExecutor.execute
// break point can find something to stop at instead of moving on to the
// next real command.
this.execPromise_.then(() => {
return browserUnderDebug.executeScriptWithDescription('', 'empty debugger hook');
});
},

// Execute a piece of code.
Expand All @@ -159,7 +162,12 @@ export class DebugHelper {
let execFn_ = () => {
// Run code through vm so that we can maintain a local scope which is
// isolated from the rest of the execution.
let res = vm_.runInContext(code, sandbox);
let res;
try {
res = vm_.runInContext(code, sandbox);
} catch (e) {
res = 'Error while evaluating command: ' + e;
}
if (!wdpromise.isPromise(res)) {
res = wdpromise.fulfilled(res);
}
Expand Down Expand Up @@ -190,14 +198,14 @@ export class DebugHelper {
deferred.fulfill(JSON.stringify(res));
}
});
return deferred;
return deferred.promise;
};
this.execute_(execFn_);
},

// Code finished executing.
resultReady: function() {
return !this.execPromise_.isPending();
return !(this.execPromise_.state_ === 'pending');
},

// Get asynchronous results synchronously.
Expand All @@ -213,7 +221,7 @@ export class DebugHelper {
}
};

return pausePromise;
return executePromise;
}

/**
Expand All @@ -227,7 +235,7 @@ export class DebugHelper {
* is done. The promise will resolve to a boolean which represents whether
* this is the first time that the debugger is called.
*/
private validatePortAvailability_(port: number): wdpromise.Promise<any> {
private validatePortAvailability_(port: number): wdpromise.Promise<boolean> {
if (this.debuggerValidated_) {
return wdpromise.fulfilled(false);
}
Expand Down Expand Up @@ -256,8 +264,9 @@ export class DebugHelper {
});

return doneDeferred.promise.then(
() => {
(firstTime: boolean) => {
this.debuggerValidated_ = true;
return firstTime;
},
(err: string) => {
console.error(err);
Expand Down
22 changes: 17 additions & 5 deletions lib/debugger/clients/explorer.js
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,10 @@ WdRepl.prototype.initServer_ = function(port) {
// Intentionally blank.
});
sock.end();
// TODO(juliemr): Investigate why this is necessary. At this point, there
// should be no active listeners so this process should just exit
// by itself.
process.exit(0);
} else if (input[input.length - 1] === '\t') {
// If the last character is the TAB key, this is an autocomplete
// request. We use everything before the TAB as the init data to feed
Expand Down Expand Up @@ -98,15 +102,17 @@ WdRepl.prototype.initRepl_ = function() {
output: process.stdout,
eval: stepEval,
useGlobal: false,
ignoreUndefined: true
ignoreUndefined: true,
completer: cmdRepl.complete.bind(cmdRepl)
});

replServer.complete = cmdRepl.complete.bind(cmdRepl);

replServer.on('exit', function() {
console.log('Exiting...');
console.log('Element Explorer Exiting...');
self.client.req({command: 'disconnect'}, function() {
// Intentionally blank.
// TODO(juliemr): Investigate why this is necessary. At this point, there
// should be no active listeners so this process should just exit
// by itself.
process.exit(0);
});
});
};
Expand Down Expand Up @@ -137,6 +143,12 @@ WdRepl.prototype.init = function() {
var self = this;
this.client = debuggerCommons.attachDebugger(process.argv[2], process.argv[3]);
this.client.once('ready', function() {
debuggerCommons.setEvaluateBreakpoint(self.client, function() {
process.send('ready');
self.client.reqContinue(function() {
// Intentionally blank.
});
});
self.initReplOrServer_();
});
};
Expand Down
Loading

0 comments on commit 530ca59

Please sign in to comment.