| /* |
| * Copyright (C) 2010 Google Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are |
| * met: |
| * |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following disclaimer |
| * in the documentation and/or other materials provided with the |
| * distribution. |
| * * Neither the name of Google Inc. nor the names of its |
| * contributors may be used to endorse or promote products derived from |
| * this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.SDKModel} |
| * @param {!WebInspector.Target} target |
| */ |
| WebInspector.DebuggerModel = function(target) |
| { |
| WebInspector.SDKModel.call(this, WebInspector.DebuggerModel, target); |
| |
| target.registerDebuggerDispatcher(new WebInspector.DebuggerDispatcher(this)); |
| this._agent = target.debuggerAgent(); |
| |
| /** @type {?WebInspector.DebuggerPausedDetails} */ |
| this._debuggerPausedDetails = null; |
| /** @type {!Object.<string, !WebInspector.Script>} */ |
| this._scripts = {}; |
| /** @type {!StringMap.<!Array.<!WebInspector.Script>>} */ |
| this._scriptsBySourceURL = new StringMap(); |
| |
| /** @type {!WebInspector.Object} */ |
| this._breakpointResolvedEventTarget = new WebInspector.Object(); |
| |
| this._isPausing = false; |
| WebInspector.settings.pauseOnExceptionEnabled.addChangeListener(this._pauseOnExceptionStateChanged, this); |
| WebInspector.settings.pauseOnCaughtException.addChangeListener(this._pauseOnExceptionStateChanged, this); |
| |
| WebInspector.settings.enableAsyncStackTraces.addChangeListener(this._asyncStackTracesStateChanged, this); |
| WebInspector.profilingLock().addEventListener(WebInspector.Lock.Events.StateChanged, this._profilingStateChanged, this); |
| |
| this.enableDebugger(); |
| |
| WebInspector.settings.skipStackFramesPattern.addChangeListener(this._applySkipStackFrameSettings, this); |
| WebInspector.settings.skipContentScripts.addChangeListener(this._applySkipStackFrameSettings, this); |
| this._applySkipStackFrameSettings(); |
| } |
| |
| /** @typedef {{location: ?WebInspector.DebuggerModel.Location, sourceURL: ?string, functionName: string, scopeChain: (Array.<!DebuggerAgent.Scope>|null)}} */ |
| WebInspector.DebuggerModel.FunctionDetails; |
| |
| /** |
| * Keep these in sync with WebCore::ScriptDebugServer |
| * |
| * @enum {string} |
| */ |
| WebInspector.DebuggerModel.PauseOnExceptionsState = { |
| DontPauseOnExceptions : "none", |
| PauseOnAllExceptions : "all", |
| PauseOnUncaughtExceptions: "uncaught" |
| }; |
| |
| WebInspector.DebuggerModel.Events = { |
| DebuggerWasEnabled: "DebuggerWasEnabled", |
| DebuggerWasDisabled: "DebuggerWasDisabled", |
| DebuggerPaused: "DebuggerPaused", |
| DebuggerResumed: "DebuggerResumed", |
| ParsedScriptSource: "ParsedScriptSource", |
| FailedToParseScriptSource: "FailedToParseScriptSource", |
| GlobalObjectCleared: "GlobalObjectCleared", |
| CallFrameSelected: "CallFrameSelected", |
| ConsoleCommandEvaluatedInSelectedCallFrame: "ConsoleCommandEvaluatedInSelectedCallFrame", |
| } |
| |
| WebInspector.DebuggerModel.BreakReason = { |
| DOM: "DOM", |
| EventListener: "EventListener", |
| XHR: "XHR", |
| Exception: "exception", |
| Assert: "assert", |
| CSPViolation: "CSPViolation", |
| DebugCommand: "debugCommand" |
| } |
| |
| WebInspector.DebuggerModel.prototype = { |
| /** |
| * @return {boolean} |
| */ |
| debuggerEnabled: function() |
| { |
| return !!this._debuggerEnabled; |
| }, |
| |
| enableDebugger: function() |
| { |
| if (this._debuggerEnabled) |
| return; |
| this._agent.enable(); |
| this._debuggerEnabled = true; |
| this._pauseOnExceptionStateChanged(); |
| this._asyncStackTracesStateChanged(); |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.DebuggerWasEnabled); |
| }, |
| |
| disableDebugger: function() |
| { |
| if (!this._debuggerEnabled) |
| return; |
| |
| this._agent.disable(); |
| this._debuggerEnabled = false; |
| this._isPausing = false; |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.DebuggerWasDisabled); |
| }, |
| |
| /** |
| * @param {boolean} skip |
| * @param {boolean=} untilReload |
| */ |
| skipAllPauses: function(skip, untilReload) |
| { |
| if (this._skipAllPausesTimeout) { |
| clearTimeout(this._skipAllPausesTimeout); |
| delete this._skipAllPausesTimeout; |
| } |
| this._agent.setSkipAllPauses(skip, untilReload); |
| }, |
| |
| /** |
| * @param {number} timeout |
| */ |
| skipAllPausesUntilReloadOrTimeout: function(timeout) |
| { |
| if (this._skipAllPausesTimeout) |
| clearTimeout(this._skipAllPausesTimeout); |
| this._agent.setSkipAllPauses(true, true); |
| // If reload happens before the timeout, the flag will be already unset and the timeout callback won't change anything. |
| this._skipAllPausesTimeout = setTimeout(this.skipAllPauses.bind(this, false), timeout); |
| }, |
| |
| _pauseOnExceptionStateChanged: function() |
| { |
| var state; |
| if (!WebInspector.settings.pauseOnExceptionEnabled.get()) { |
| state = WebInspector.DebuggerModel.PauseOnExceptionsState.DontPauseOnExceptions; |
| } else if (WebInspector.settings.pauseOnCaughtException.get()) { |
| state = WebInspector.DebuggerModel.PauseOnExceptionsState.PauseOnAllExceptions; |
| } else { |
| state = WebInspector.DebuggerModel.PauseOnExceptionsState.PauseOnUncaughtExceptions; |
| } |
| this._agent.setPauseOnExceptions(state); |
| }, |
| |
| _profilingStateChanged: function() |
| { |
| if (WebInspector.experimentsSettings.disableAgentsWhenProfile.isEnabled()) { |
| if (WebInspector.profilingLock().isAcquired()) |
| this.disableDebugger(); |
| else |
| this.enableDebugger(); |
| } |
| this._asyncStackTracesStateChanged(); |
| }, |
| |
| _asyncStackTracesStateChanged: function() |
| { |
| const maxAsyncStackChainDepth = 4; |
| var enabled = WebInspector.settings.enableAsyncStackTraces.get() && !WebInspector.profilingLock().isAcquired(); |
| this._agent.setAsyncCallStackDepth(enabled ? maxAsyncStackChainDepth : 0); |
| }, |
| |
| stepInto: function() |
| { |
| /** |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function callback() |
| { |
| this._agent.stepInto(); |
| } |
| this._agent.setOverlayMessage(undefined, callback.bind(this)); |
| }, |
| |
| stepOver: function() |
| { |
| /** |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function callback() |
| { |
| this._agent.stepOver(); |
| } |
| this._agent.setOverlayMessage(undefined, callback.bind(this)); |
| }, |
| |
| stepOut: function() |
| { |
| /** |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function callback() |
| { |
| this._agent.stepOut(); |
| } |
| this._agent.setOverlayMessage(undefined, callback.bind(this)); |
| }, |
| |
| resume: function() |
| { |
| /** |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function callback() |
| { |
| this._agent.resume(); |
| } |
| this._agent.setOverlayMessage(undefined, callback.bind(this)); |
| this._isPausing = false; |
| }, |
| |
| pause: function() |
| { |
| this._isPausing = true; |
| this.skipAllPauses(false); |
| this._agent.pause(); |
| }, |
| |
| /** |
| * @param {string} url |
| * @param {number} lineNumber |
| * @param {number=} columnNumber |
| * @param {string=} condition |
| * @param {function(?DebuggerAgent.BreakpointId, !Array.<!WebInspector.DebuggerModel.Location>)=} callback |
| */ |
| setBreakpointByURL: function(url, lineNumber, columnNumber, condition, callback) |
| { |
| // Adjust column if needed. |
| var minColumnNumber = 0; |
| var scripts = this._scriptsBySourceURL.get(url) || []; |
| for (var i = 0, l = scripts.length; i < l; ++i) { |
| var script = scripts[i]; |
| if (lineNumber === script.lineOffset) |
| minColumnNumber = minColumnNumber ? Math.min(minColumnNumber, script.columnOffset) : script.columnOffset; |
| } |
| columnNumber = Math.max(columnNumber, minColumnNumber); |
| |
| var target = this.target(); |
| /** |
| * @param {?Protocol.Error} error |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {!Array.<!DebuggerAgent.Location>} locations |
| */ |
| function didSetBreakpoint(error, breakpointId, locations) |
| { |
| if (callback) { |
| var rawLocations = locations ? locations.map(WebInspector.DebuggerModel.Location.fromPayload.bind(WebInspector.DebuggerModel.Location, target)) : []; |
| callback(error ? null : breakpointId, rawLocations); |
| } |
| } |
| this._agent.setBreakpointByUrl(lineNumber, url, undefined, columnNumber, condition, undefined, didSetBreakpoint); |
| WebInspector.userMetrics.ScriptsBreakpointSet.record(); |
| }, |
| |
| /** |
| * @param {!WebInspector.DebuggerModel.Location} rawLocation |
| * @param {string} condition |
| * @param {function(?DebuggerAgent.BreakpointId, !Array.<!WebInspector.DebuggerModel.Location>)=} callback |
| */ |
| setBreakpointBySourceId: function(rawLocation, condition, callback) |
| { |
| var target = this.target(); |
| |
| /** |
| * @param {?Protocol.Error} error |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {!DebuggerAgent.Location} actualLocation |
| */ |
| function didSetBreakpoint(error, breakpointId, actualLocation) |
| { |
| if (callback) { |
| var location = WebInspector.DebuggerModel.Location.fromPayload(target, actualLocation); |
| callback(error ? null : breakpointId, [location]); |
| } |
| } |
| this._agent.setBreakpoint(rawLocation.payload(), condition, didSetBreakpoint); |
| WebInspector.userMetrics.ScriptsBreakpointSet.record(); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {function()=} callback |
| */ |
| removeBreakpoint: function(breakpointId, callback) |
| { |
| this._agent.removeBreakpoint(breakpointId, innerCallback); |
| |
| /** |
| * @param {?Protocol.Error} error |
| */ |
| function innerCallback(error) |
| { |
| if (error) |
| console.error("Failed to remove breakpoint: " + error); |
| if (callback) |
| callback(); |
| } |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {!DebuggerAgent.Location} location |
| */ |
| _breakpointResolved: function(breakpointId, location) |
| { |
| this._breakpointResolvedEventTarget.dispatchEventToListeners(breakpointId, WebInspector.DebuggerModel.Location.fromPayload(this.target(), location)); |
| }, |
| |
| _globalObjectCleared: function() |
| { |
| this._setDebuggerPausedDetails(null); |
| this._reset(); |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.GlobalObjectCleared); |
| }, |
| |
| _reset: function() |
| { |
| this._scripts = {}; |
| this._scriptsBySourceURL.clear(); |
| }, |
| |
| /** |
| * @return {!Object.<string, !WebInspector.Script>} |
| */ |
| get scripts() |
| { |
| return this._scripts; |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @return {!WebInspector.Script} |
| */ |
| scriptForId: function(scriptId) |
| { |
| return this._scripts[scriptId] || null; |
| }, |
| |
| /** |
| * @return {!Array.<!WebInspector.Script>} |
| */ |
| scriptsForSourceURL: function(sourceURL) |
| { |
| if (!sourceURL) |
| return []; |
| return this._scriptsBySourceURL.get(sourceURL) || []; |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @param {string} newSource |
| * @param {function(?Protocol.Error, !DebuggerAgent.SetScriptSourceError=)} callback |
| */ |
| setScriptSource: function(scriptId, newSource, callback) |
| { |
| this._scripts[scriptId].editSource(newSource, this._didEditScriptSource.bind(this, scriptId, newSource, callback)); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @param {string} newSource |
| * @param {function(?Protocol.Error, !DebuggerAgent.SetScriptSourceError=)} callback |
| * @param {?Protocol.Error} error |
| * @param {!DebuggerAgent.SetScriptSourceError=} errorData |
| * @param {!Array.<!DebuggerAgent.CallFrame>=} callFrames |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| * @param {boolean=} needsStepIn |
| */ |
| _didEditScriptSource: function(scriptId, newSource, callback, error, errorData, callFrames, asyncStackTrace, needsStepIn) |
| { |
| if (needsStepIn) { |
| this.stepInto(); |
| this._pendingLiveEditCallback = callback.bind(this, error, errorData); |
| return; |
| } |
| |
| if (!error && callFrames && callFrames.length) |
| this._pausedScript(callFrames, this._debuggerPausedDetails.reason, this._debuggerPausedDetails.auxData, this._debuggerPausedDetails.breakpointIds, asyncStackTrace); |
| callback(error, errorData); |
| }, |
| |
| /** |
| * @return {?Array.<!WebInspector.DebuggerModel.CallFrame>} |
| */ |
| get callFrames() |
| { |
| return this._debuggerPausedDetails ? this._debuggerPausedDetails.callFrames : null; |
| }, |
| |
| /** |
| * @return {?WebInspector.DebuggerPausedDetails} |
| */ |
| debuggerPausedDetails: function() |
| { |
| return this._debuggerPausedDetails; |
| }, |
| |
| /** |
| * @param {?WebInspector.DebuggerPausedDetails} debuggerPausedDetails |
| */ |
| _setDebuggerPausedDetails: function(debuggerPausedDetails) |
| { |
| this._isPausing = false; |
| this._debuggerPausedDetails = debuggerPausedDetails; |
| if (this._debuggerPausedDetails) |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.DebuggerPaused, this._debuggerPausedDetails); |
| if (debuggerPausedDetails) { |
| this.setSelectedCallFrame(debuggerPausedDetails.callFrames[0]); |
| this._agent.setOverlayMessage(WebInspector.UIString("Paused in debugger")); |
| } else { |
| this.setSelectedCallFrame(null); |
| this._agent.setOverlayMessage(); |
| } |
| }, |
| |
| /** |
| * @param {!Array.<!DebuggerAgent.CallFrame>} callFrames |
| * @param {string} reason |
| * @param {!Object|undefined} auxData |
| * @param {!Array.<string>} breakpointIds |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| */ |
| _pausedScript: function(callFrames, reason, auxData, breakpointIds, asyncStackTrace) |
| { |
| this._setDebuggerPausedDetails(new WebInspector.DebuggerPausedDetails(this.target(), callFrames, reason, auxData, breakpointIds, asyncStackTrace)); |
| if (this._pendingLiveEditCallback) { |
| var callback = this._pendingLiveEditCallback; |
| delete this._pendingLiveEditCallback; |
| callback(); |
| } |
| }, |
| |
| _resumedScript: function() |
| { |
| this._setDebuggerPausedDetails(null); |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.DebuggerResumed); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @param {string} sourceURL |
| * @param {number} startLine |
| * @param {number} startColumn |
| * @param {number} endLine |
| * @param {number} endColumn |
| * @param {boolean} isContentScript |
| * @param {string=} sourceMapURL |
| * @param {boolean=} hasSourceURL |
| * @param {boolean=} hasSyntaxError |
| */ |
| _parsedScriptSource: function(scriptId, sourceURL, startLine, startColumn, endLine, endColumn, isContentScript, sourceMapURL, hasSourceURL, hasSyntaxError) |
| { |
| var script = new WebInspector.Script(this.target(), scriptId, sourceURL, startLine, startColumn, endLine, endColumn, isContentScript, sourceMapURL, hasSourceURL); |
| this._registerScript(script); |
| if (!hasSyntaxError) |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.ParsedScriptSource, script); |
| else |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.FailedToParseScriptSource, script); |
| }, |
| |
| /** |
| * @param {!WebInspector.Script} script |
| */ |
| _registerScript: function(script) |
| { |
| this._scripts[script.scriptId] = script; |
| if (script.isAnonymousScript()) |
| return; |
| |
| var scripts = this._scriptsBySourceURL.get(script.sourceURL); |
| if (!scripts) { |
| scripts = []; |
| this._scriptsBySourceURL.set(script.sourceURL, scripts); |
| } |
| scripts.push(script); |
| }, |
| |
| /** |
| * @param {!WebInspector.Script} script |
| * @param {number} lineNumber |
| * @param {number} columnNumber |
| * @return {?WebInspector.DebuggerModel.Location} |
| */ |
| createRawLocation: function(script, lineNumber, columnNumber) |
| { |
| if (script.sourceURL) |
| return this.createRawLocationByURL(script.sourceURL, lineNumber, columnNumber); |
| return new WebInspector.DebuggerModel.Location(this.target(), script.scriptId, lineNumber, columnNumber); |
| }, |
| |
| /** |
| * @param {string} sourceURL |
| * @param {number} lineNumber |
| * @param {number} columnNumber |
| * @return {?WebInspector.DebuggerModel.Location} |
| */ |
| createRawLocationByURL: function(sourceURL, lineNumber, columnNumber) |
| { |
| var closestScript = null; |
| var scripts = this._scriptsBySourceURL.get(sourceURL) || []; |
| for (var i = 0, l = scripts.length; i < l; ++i) { |
| var script = scripts[i]; |
| if (!closestScript) |
| closestScript = script; |
| if (script.lineOffset > lineNumber || (script.lineOffset === lineNumber && script.columnOffset > columnNumber)) |
| continue; |
| if (script.endLine < lineNumber || (script.endLine === lineNumber && script.endColumn <= columnNumber)) |
| continue; |
| closestScript = script; |
| break; |
| } |
| return closestScript ? new WebInspector.DebuggerModel.Location(this.target(), closestScript.scriptId, lineNumber, columnNumber) : null; |
| }, |
| |
| /** |
| * @param {?DebuggerAgent.ScriptId} scriptId |
| * @param {string} sourceUrl |
| * @param {number} lineNumber |
| * @param {number} columnNumber |
| * @return {?WebInspector.DebuggerModel.Location} |
| */ |
| createRawLocationByScriptId: function(scriptId, sourceUrl, lineNumber, columnNumber) |
| { |
| var script = scriptId ? this.scriptForId(scriptId) : null; |
| return script ? this.createRawLocation(script, lineNumber, columnNumber) : this.createRawLocationByURL(sourceUrl, lineNumber, columnNumber); |
| }, |
| |
| /** |
| * @return {boolean} |
| */ |
| isPaused: function() |
| { |
| return !!this.debuggerPausedDetails(); |
| }, |
| |
| /** |
| * @return {boolean} |
| */ |
| isPausing: function() |
| { |
| return this._isPausing; |
| }, |
| |
| /** |
| * @param {?WebInspector.DebuggerModel.CallFrame} callFrame |
| */ |
| setSelectedCallFrame: function(callFrame) |
| { |
| this._selectedCallFrame = callFrame; |
| if (!this._selectedCallFrame) |
| return; |
| |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.CallFrameSelected, callFrame); |
| }, |
| |
| /** |
| * @return {?WebInspector.DebuggerModel.CallFrame} |
| */ |
| selectedCallFrame: function() |
| { |
| return this._selectedCallFrame; |
| }, |
| |
| /** |
| * @param {string} code |
| * @param {string} objectGroup |
| * @param {boolean} includeCommandLineAPI |
| * @param {boolean} doNotPauseOnExceptionsAndMuteConsole |
| * @param {boolean} returnByValue |
| * @param {boolean} generatePreview |
| * @param {function(?WebInspector.RemoteObject, boolean, ?RuntimeAgent.RemoteObject=, ?DebuggerAgent.ExceptionDetails=)} callback |
| */ |
| evaluateOnSelectedCallFrame: function(code, objectGroup, includeCommandLineAPI, doNotPauseOnExceptionsAndMuteConsole, returnByValue, generatePreview, callback) |
| { |
| /** |
| * @param {?RuntimeAgent.RemoteObject} result |
| * @param {boolean=} wasThrown |
| * @param {?DebuggerAgent.ExceptionDetails=} exceptionDetails |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function didEvaluate(result, wasThrown, exceptionDetails) |
| { |
| if (!result) |
| callback(null, false); |
| else if (returnByValue) |
| callback(null, !!wasThrown, wasThrown ? null : result, exceptionDetails); |
| else |
| callback(this.target().runtimeModel.createRemoteObject(result), !!wasThrown, undefined, exceptionDetails); |
| |
| if (objectGroup === "console") |
| this.dispatchEventToListeners(WebInspector.DebuggerModel.Events.ConsoleCommandEvaluatedInSelectedCallFrame); |
| } |
| |
| this.selectedCallFrame().evaluate(code, objectGroup, includeCommandLineAPI, doNotPauseOnExceptionsAndMuteConsole, returnByValue, generatePreview, didEvaluate.bind(this)); |
| }, |
| |
| /** |
| * @param {function(!Object)} callback |
| */ |
| getSelectedCallFrameVariables: function(callback) |
| { |
| var result = { this: true }; |
| |
| var selectedCallFrame = this._selectedCallFrame; |
| if (!selectedCallFrame) |
| callback(result); |
| |
| var pendingRequests = 0; |
| |
| function propertiesCollected(properties) |
| { |
| for (var i = 0; properties && i < properties.length; ++i) |
| result[properties[i].name] = true; |
| if (--pendingRequests == 0) |
| callback(result); |
| } |
| |
| for (var i = 0; i < selectedCallFrame.scopeChain.length; ++i) { |
| var scope = selectedCallFrame.scopeChain[i]; |
| var object = this.target().runtimeModel.createRemoteObject(scope.object); |
| pendingRequests++; |
| object.getAllProperties(false, propertiesCollected); |
| } |
| }, |
| |
| /** |
| * Handles notification from JavaScript VM about updated stack (liveedit or frame restart action). |
| * @param {!Array.<!DebuggerAgent.CallFrame>=} newCallFrames |
| * @param {!Object=} details |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| */ |
| callStackModified: function(newCallFrames, details, asyncStackTrace) |
| { |
| // FIXME: declare this property in protocol and in JavaScript. |
| if (details && details["stack_update_needs_step_in"]) |
| this.stepInto(); |
| else if (newCallFrames && newCallFrames.length) |
| this._pausedScript(newCallFrames, this._debuggerPausedDetails.reason, this._debuggerPausedDetails.auxData, this._debuggerPausedDetails.breakpointIds, asyncStackTrace); |
| }, |
| |
| _applySkipStackFrameSettings: function() |
| { |
| this._agent.skipStackFrames(WebInspector.settings.skipStackFramesPattern.get(), WebInspector.settings.skipContentScripts.get()); |
| }, |
| |
| /** |
| * @param {!WebInspector.RemoteObject} remoteObject |
| * @param {function(?WebInspector.DebuggerModel.FunctionDetails)} callback |
| */ |
| functionDetails: function(remoteObject, callback) |
| { |
| this._agent.getFunctionDetails(remoteObject.objectId, didGetDetails.bind(this)); |
| |
| /** |
| * @param {?Protocol.Error} error |
| * @param {!DebuggerAgent.FunctionDetails} response |
| * @this {WebInspector.DebuggerModel} |
| */ |
| function didGetDetails(error, response) |
| { |
| if (error) { |
| console.error(error); |
| callback(null); |
| return; |
| } |
| var location = response.location; |
| var script = this.scriptForId(location.scriptId); |
| var rawLocation = script ? this.createRawLocation(script, location.lineNumber + 1, location.columnNumber + 1) : null; |
| var sourceURL = script ? script.contentURL() : null; |
| callback({location: rawLocation, sourceURL: sourceURL, functionName: response.functionName, scopeChain: response.scopeChain || null}); |
| } |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {function(!WebInspector.Event)} listener |
| * @param {!Object=} thisObject |
| */ |
| addBreakpointListener: function(breakpointId, listener, thisObject) |
| { |
| this._breakpointResolvedEventTarget.addEventListener(breakpointId, listener, thisObject) |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {function(!WebInspector.Event)} listener |
| * @param {!Object=} thisObject |
| */ |
| removeBreakpointListener: function(breakpointId, listener, thisObject) |
| { |
| this._breakpointResolvedEventTarget.removeEventListener(breakpointId, listener, thisObject); |
| }, |
| |
| dispose: function() |
| { |
| WebInspector.settings.pauseOnExceptionEnabled.removeChangeListener(this._pauseOnExceptionStateChanged, this); |
| WebInspector.settings.pauseOnCaughtException.removeChangeListener(this._pauseOnExceptionStateChanged, this); |
| WebInspector.settings.skipStackFramesPattern.removeChangeListener(this._applySkipStackFrameSettings, this); |
| WebInspector.settings.skipContentScripts.removeChangeListener(this._applySkipStackFrameSettings, this); |
| WebInspector.settings.enableAsyncStackTraces.removeChangeListener(this._asyncStackTracesStateChanged, this); |
| }, |
| |
| __proto__: WebInspector.SDKModel.prototype |
| } |
| |
| WebInspector.DebuggerEventTypes = { |
| JavaScriptPause: 0, |
| JavaScriptBreakpoint: 1, |
| NativeBreakpoint: 2 |
| }; |
| |
| /** |
| * @constructor |
| * @implements {DebuggerAgent.Dispatcher} |
| * @param {!WebInspector.DebuggerModel} debuggerModel |
| */ |
| WebInspector.DebuggerDispatcher = function(debuggerModel) |
| { |
| this._debuggerModel = debuggerModel; |
| } |
| |
| WebInspector.DebuggerDispatcher.prototype = { |
| /** |
| * @param {!Array.<!DebuggerAgent.CallFrame>} callFrames |
| * @param {string} reason |
| * @param {!Object=} auxData |
| * @param {!Array.<string>=} breakpointIds |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| */ |
| paused: function(callFrames, reason, auxData, breakpointIds, asyncStackTrace) |
| { |
| this._debuggerModel._pausedScript(callFrames, reason, auxData, breakpointIds || [], asyncStackTrace); |
| }, |
| |
| /** |
| * @override |
| */ |
| resumed: function() |
| { |
| this._debuggerModel._resumedScript(); |
| }, |
| |
| /** |
| * @override |
| */ |
| globalObjectCleared: function() |
| { |
| this._debuggerModel._globalObjectCleared(); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @param {string} sourceURL |
| * @param {number} startLine |
| * @param {number} startColumn |
| * @param {number} endLine |
| * @param {number} endColumn |
| * @param {boolean=} isContentScript |
| * @param {string=} sourceMapURL |
| * @param {boolean=} hasSourceURL |
| */ |
| scriptParsed: function(scriptId, sourceURL, startLine, startColumn, endLine, endColumn, isContentScript, sourceMapURL, hasSourceURL) |
| { |
| this._debuggerModel._parsedScriptSource(scriptId, sourceURL, startLine, startColumn, endLine, endColumn, !!isContentScript, sourceMapURL, hasSourceURL, false); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.ScriptId} scriptId |
| * @param {string} sourceURL |
| * @param {number} startLine |
| * @param {number} startColumn |
| * @param {number} endLine |
| * @param {number} endColumn |
| * @param {boolean=} isContentScript |
| * @param {string=} sourceMapURL |
| * @param {boolean=} hasSourceURL |
| */ |
| scriptFailedToParse: function(scriptId, sourceURL, startLine, startColumn, endLine, endColumn, isContentScript, sourceMapURL, hasSourceURL) |
| { |
| this._debuggerModel._parsedScriptSource(scriptId, sourceURL, startLine, startColumn, endLine, endColumn, !!isContentScript, sourceMapURL, hasSourceURL, true); |
| }, |
| |
| /** |
| * @param {!DebuggerAgent.BreakpointId} breakpointId |
| * @param {!DebuggerAgent.Location} location |
| */ |
| breakpointResolved: function(breakpointId, location) |
| { |
| this._debuggerModel._breakpointResolved(breakpointId, location); |
| } |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.SDKObject} |
| * @param {!WebInspector.Target} target |
| * @param {string} scriptId |
| * @param {number} lineNumber |
| * @param {number=} columnNumber |
| */ |
| WebInspector.DebuggerModel.Location = function(target, scriptId, lineNumber, columnNumber) |
| { |
| WebInspector.SDKObject.call(this, target); |
| this._debuggerModel = target.debuggerModel; |
| this.scriptId = scriptId; |
| this.lineNumber = lineNumber; |
| this.columnNumber = columnNumber || 0; |
| } |
| |
| /** |
| * @param {!WebInspector.Target} target |
| * @param {!DebuggerAgent.Location} payload |
| * @return {!WebInspector.DebuggerModel.Location} |
| */ |
| WebInspector.DebuggerModel.Location.fromPayload = function(target, payload) |
| { |
| return new WebInspector.DebuggerModel.Location(target, payload.scriptId, payload.lineNumber, payload.columnNumber); |
| } |
| |
| WebInspector.DebuggerModel.Location.prototype = { |
| /** |
| * @return {!DebuggerAgent.Location} |
| */ |
| payload: function() |
| { |
| return { scriptId: this.scriptId, lineNumber: this.lineNumber, columnNumber: this.columnNumber }; |
| }, |
| |
| /** |
| * @return {!WebInspector.Script} |
| */ |
| script: function() |
| { |
| return this._debuggerModel.scriptForId(this.scriptId); |
| }, |
| |
| continueToLocation: function() |
| { |
| this._debuggerModel._agent.continueToLocation(this.payload()); |
| }, |
| |
| /** |
| * @return {string} |
| */ |
| id: function() |
| { |
| return this.target().id() + ":" + this.scriptId + ":" + this.lineNumber + ":" + this.columnNumber |
| }, |
| |
| __proto__: WebInspector.SDKObject.prototype |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.SDKObject} |
| * @param {!WebInspector.Target} target |
| * @param {!WebInspector.Script} script |
| * @param {!DebuggerAgent.CallFrame} payload |
| * @param {boolean=} isAsync |
| */ |
| WebInspector.DebuggerModel.CallFrame = function(target, script, payload, isAsync) |
| { |
| WebInspector.SDKObject.call(this, target); |
| this._debuggerAgent = target.debuggerModel._agent; |
| this._script = script; |
| this._payload = payload; |
| this._isAsync = isAsync; |
| this._location = WebInspector.DebuggerModel.Location.fromPayload(target, payload.location); |
| } |
| |
| /** |
| * @param {!WebInspector.Target} target |
| * @param {!Array.<!DebuggerAgent.CallFrame>} callFrames |
| * @param {boolean=} isAsync |
| * @return {!Array.<!WebInspector.DebuggerModel.CallFrame>} |
| */ |
| WebInspector.DebuggerModel.CallFrame.fromPayloadArray = function(target, callFrames, isAsync) |
| { |
| var result = []; |
| for (var i = 0; i < callFrames.length; ++i) { |
| var callFrame = callFrames[i]; |
| var script = target.debuggerModel.scriptForId(callFrame.location.scriptId); |
| if (script) |
| result.push(new WebInspector.DebuggerModel.CallFrame(target, script, callFrame, isAsync)); |
| } |
| return result; |
| } |
| |
| WebInspector.DebuggerModel.CallFrame.prototype = { |
| |
| /** |
| * @return {!WebInspector.Script} |
| */ |
| get script() |
| { |
| return this._script; |
| }, |
| |
| /** |
| * @return {string} |
| */ |
| get type() |
| { |
| return this._payload.type; |
| }, |
| |
| /** |
| * @return {string} |
| */ |
| get id() |
| { |
| return this._payload.callFrameId; |
| }, |
| |
| /** |
| * @return {!Array.<!DebuggerAgent.Scope>} |
| */ |
| get scopeChain() |
| { |
| return this._payload.scopeChain; |
| }, |
| |
| /** |
| * @return {?WebInspector.RemoteObject} |
| */ |
| thisObject: function() |
| { |
| return this._payload.this ? this.target().runtimeModel.createRemoteObject(this._payload.this) : null; |
| }, |
| |
| /** |
| * @return {?WebInspector.RemoteObject} |
| */ |
| returnValue: function() |
| { |
| return this._payload.returnValue ? this.target().runtimeModel.createRemoteObject(this._payload.returnValue) : null |
| }, |
| |
| /** |
| * @return {string} |
| */ |
| get functionName() |
| { |
| return this._payload.functionName; |
| }, |
| |
| /** |
| * @return {!WebInspector.DebuggerModel.Location} |
| */ |
| location: function() |
| { |
| return this._location; |
| }, |
| |
| /** |
| * @return {boolean} |
| */ |
| isAsync: function() |
| { |
| return !!this._isAsync; |
| }, |
| |
| /** |
| * @param {string} code |
| * @param {string} objectGroup |
| * @param {boolean} includeCommandLineAPI |
| * @param {boolean} doNotPauseOnExceptionsAndMuteConsole |
| * @param {boolean} returnByValue |
| * @param {boolean} generatePreview |
| * @param {function(?RuntimeAgent.RemoteObject, boolean=, ?DebuggerAgent.ExceptionDetails=)} callback |
| */ |
| evaluate: function(code, objectGroup, includeCommandLineAPI, doNotPauseOnExceptionsAndMuteConsole, returnByValue, generatePreview, callback) |
| { |
| /** |
| * @param {?Protocol.Error} error |
| * @param {!RuntimeAgent.RemoteObject} result |
| * @param {boolean=} wasThrown |
| * @param {?DebuggerAgent.ExceptionDetails=} exceptionDetails |
| */ |
| function didEvaluateOnCallFrame(error, result, wasThrown, exceptionDetails) |
| { |
| if (error) { |
| console.error(error); |
| callback(null, false); |
| return; |
| } |
| callback(result, wasThrown, exceptionDetails); |
| } |
| this._debuggerAgent.evaluateOnCallFrame(this._payload.callFrameId, code, objectGroup, includeCommandLineAPI, doNotPauseOnExceptionsAndMuteConsole, returnByValue, generatePreview, didEvaluateOnCallFrame); |
| }, |
| |
| /** |
| * @param {function(?Protocol.Error=)=} callback |
| */ |
| restart: function(callback) |
| { |
| /** |
| * @param {?Protocol.Error} error |
| * @param {!Array.<!DebuggerAgent.CallFrame>=} callFrames |
| * @param {!Object=} details |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| * @this {WebInspector.DebuggerModel.CallFrame} |
| */ |
| function protocolCallback(error, callFrames, details, asyncStackTrace) |
| { |
| if (!error) |
| this.target().debuggerModel.callStackModified(callFrames, details, asyncStackTrace); |
| if (callback) |
| callback(error); |
| } |
| this._debuggerAgent.restartFrame(this._payload.callFrameId, protocolCallback.bind(this)); |
| }, |
| |
| __proto__: WebInspector.SDKObject.prototype |
| } |
| |
| /** |
| * @constructor |
| * @param {!Array.<!WebInspector.DebuggerModel.CallFrame>} callFrames |
| * @param {?WebInspector.DebuggerModel.StackTrace} asyncStackTrace |
| * @param {string=} description |
| */ |
| WebInspector.DebuggerModel.StackTrace = function(callFrames, asyncStackTrace, description) |
| { |
| this.callFrames = callFrames; |
| this.asyncStackTrace = asyncStackTrace; |
| this.description = description; |
| } |
| |
| /** |
| * @param {!WebInspector.Target} target |
| * @param {!DebuggerAgent.StackTrace=} payload |
| * @param {boolean=} isAsync |
| * @return {?WebInspector.DebuggerModel.StackTrace} |
| */ |
| WebInspector.DebuggerModel.StackTrace.fromPayload = function(target, payload, isAsync) |
| { |
| if (!payload) |
| return null; |
| var callFrames = WebInspector.DebuggerModel.CallFrame.fromPayloadArray(target, payload.callFrames, isAsync); |
| if (!callFrames.length) |
| return null; |
| var asyncStackTrace = WebInspector.DebuggerModel.StackTrace.fromPayload(target, payload.asyncStackTrace, true); |
| return new WebInspector.DebuggerModel.StackTrace(callFrames, asyncStackTrace, payload.description); |
| } |
| |
| /** |
| * @constructor |
| * @extends {WebInspector.SDKObject} |
| * @param {!WebInspector.Target} target |
| * @param {!Array.<!DebuggerAgent.CallFrame>} callFrames |
| * @param {string} reason |
| * @param {!Object|undefined} auxData |
| * @param {!Array.<string>} breakpointIds |
| * @param {!DebuggerAgent.StackTrace=} asyncStackTrace |
| */ |
| WebInspector.DebuggerPausedDetails = function(target, callFrames, reason, auxData, breakpointIds, asyncStackTrace) |
| { |
| WebInspector.SDKObject.call(this, target); |
| this.callFrames = WebInspector.DebuggerModel.CallFrame.fromPayloadArray(target, callFrames); |
| this.reason = reason; |
| this.auxData = auxData; |
| this.breakpointIds = breakpointIds; |
| this.asyncStackTrace = WebInspector.DebuggerModel.StackTrace.fromPayload(target, asyncStackTrace, true); |
| } |
| |
| WebInspector.DebuggerPausedDetails.prototype = { |
| /** |
| * @return {?WebInspector.RemoteObject} |
| */ |
| exception: function() |
| { |
| if (this.reason !== WebInspector.DebuggerModel.BreakReason.Exception) |
| return null; |
| return this.target().runtimeModel.createRemoteObject(/** @type {!RuntimeAgent.RemoteObject} */(this.auxData)); |
| }, |
| |
| __proto__: WebInspector.SDKObject.prototype |
| } |
| |
| /** |
| * @type {!WebInspector.DebuggerModel} |
| */ |
| WebInspector.debuggerModel; |