UNPKG

debug-server-next

Version:

Dev server for hippy-core.

188 lines (187 loc) 8.13 kB
/* * Copyright (C) 2012 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. */ import * as SDK from '../../core/sdk/sdk.js'; import * as TextUtils from '../text_utils/text_utils.js'; import * as Workspace from '../workspace/workspace.js'; import { DebuggerWorkspaceBinding } from './DebuggerWorkspaceBinding.js'; import { LiveLocationPool } from './LiveLocation.js'; const debuggerModelToMessageHelperMap = new WeakMap(); export class PresentationConsoleMessageManager { constructor() { SDK.TargetManager.TargetManager.instance().observeModels(SDK.DebuggerModel.DebuggerModel, this); SDK.ConsoleModel.ConsoleModel.instance().addEventListener(SDK.ConsoleModel.Events.ConsoleCleared, this._consoleCleared, this); SDK.ConsoleModel.ConsoleModel.instance().addEventListener(SDK.ConsoleModel.Events.MessageAdded, event => this._consoleMessageAdded(event.data)); SDK.ConsoleModel.ConsoleModel.instance().messages().forEach(this._consoleMessageAdded, this); } modelAdded(debuggerModel) { debuggerModelToMessageHelperMap.set(debuggerModel, new PresentationConsoleMessageHelper(debuggerModel)); } modelRemoved(debuggerModel) { const helper = debuggerModelToMessageHelperMap.get(debuggerModel); if (helper) { helper._consoleCleared(); } } _consoleMessageAdded(message) { const runtimeModel = message.runtimeModel(); if (!message.isErrorOrWarning() || !message.runtimeModel() || message.source === "violation" /* Violation */ || !runtimeModel) { return; } const helper = debuggerModelToMessageHelperMap.get(runtimeModel.debuggerModel()); if (helper) { helper._consoleMessageAdded(message); } } _consoleCleared() { for (const debuggerModel of SDK.TargetManager.TargetManager.instance().models(SDK.DebuggerModel.DebuggerModel)) { const helper = debuggerModelToMessageHelperMap.get(debuggerModel); if (helper) { helper._consoleCleared(); } } } } export class PresentationConsoleMessageHelper { _debuggerModel; _pendingConsoleMessages; _presentationConsoleMessages; _locationPool; constructor(debuggerModel) { this._debuggerModel = debuggerModel; this._pendingConsoleMessages = new Map(); this._presentationConsoleMessages = []; // TODO(dgozman): queueMicrotask because we race with DebuggerWorkspaceBinding on ParsedScriptSource event delivery. debuggerModel.addEventListener(SDK.DebuggerModel.Events.ParsedScriptSource, event => { queueMicrotask(() => { this._parsedScriptSource(event); }); }); debuggerModel.addEventListener(SDK.DebuggerModel.Events.GlobalObjectCleared, this._debuggerReset, this); this._locationPool = new LiveLocationPool(); } _consoleMessageAdded(message) { const rawLocation = this._rawLocation(message); if (rawLocation) { this._addConsoleMessageToScript(message, rawLocation); } else { this._addPendingConsoleMessage(message); } } _rawLocation(message) { if (message.scriptId) { return this._debuggerModel.createRawLocationByScriptId(message.scriptId, message.line, message.column); } const callFrame = message.stackTrace && message.stackTrace.callFrames ? message.stackTrace.callFrames[0] : null; if (callFrame) { return this._debuggerModel.createRawLocationByScriptId(callFrame.scriptId, callFrame.lineNumber, callFrame.columnNumber); } if (message.url) { return this._debuggerModel.createRawLocationByURL(message.url, message.line, message.column); } return null; } _addConsoleMessageToScript(message, rawLocation) { this._presentationConsoleMessages.push(new PresentationConsoleMessage(message, rawLocation, this._locationPool)); } _addPendingConsoleMessage(message) { if (!message.url) { return; } const pendingMessages = this._pendingConsoleMessages.get(message.url); if (!pendingMessages) { this._pendingConsoleMessages.set(message.url, [message]); } else { pendingMessages.push(message); } } _parsedScriptSource(event) { const script = event.data; const messages = this._pendingConsoleMessages.get(script.sourceURL); if (!messages) { return; } const pendingMessages = []; for (const message of messages) { const rawLocation = this._rawLocation(message); if (rawLocation && script.scriptId === rawLocation.scriptId) { this._addConsoleMessageToScript(message, rawLocation); } else { pendingMessages.push(message); } } if (pendingMessages.length) { this._pendingConsoleMessages.set(script.sourceURL, pendingMessages); } else { this._pendingConsoleMessages.delete(script.sourceURL); } } _consoleCleared() { this._pendingConsoleMessages = new Map(); this._debuggerReset(); } _debuggerReset() { for (const message of this._presentationConsoleMessages) { message.dispose(); } this._presentationConsoleMessages = []; this._locationPool.disposeAll(); } } export class PresentationConsoleMessage extends Workspace.UISourceCode.Message { uiSourceCode; constructor(message, rawLocation, locationPool) { const level = message.level === "error" /* Error */ ? Workspace.UISourceCode.Message.Level.Error : Workspace.UISourceCode.Message.Level.Warning; super(level, message.messageText); DebuggerWorkspaceBinding.instance().createLiveLocation(rawLocation, this._updateLocation.bind(this), locationPool); } async _updateLocation(liveLocation) { if (this.uiSourceCode) { this.uiSourceCode.removeMessage(this); } const uiLocation = await liveLocation.uiLocation(); if (!uiLocation) { return; } this._range = TextUtils.TextRange.TextRange.createFromLocation(uiLocation.lineNumber, uiLocation.columnNumber || 0); this.uiSourceCode = uiLocation.uiSourceCode; this.uiSourceCode.addMessage(this); } dispose() { if (this.uiSourceCode) { this.uiSourceCode.removeMessage(this); } } }