UNPKG

@quick-game/cli

Version:

Command line interface for rapid qg development

432 lines 19.5 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 Common from '../../core/common/common.js'; import * as i18n from '../../core/i18n/i18n.js'; import * as Platform from '../../core/platform/platform.js'; import * as SDK from '../../core/sdk/sdk.js'; import * as Workspace from '../workspace/workspace.js'; import { ContentProviderBasedProject } from './ContentProviderBasedProject.js'; import { DebuggerWorkspaceBinding } from './DebuggerWorkspaceBinding.js'; import { NetworkProject } from './NetworkProject.js'; import { metadataForURL } from './ResourceUtils.js'; const UIStrings = { /** *@description Error text displayed in the console when editing a live script fails. LiveEdit is *the name of the feature for editing code that is already running. *@example {warning} PH1 */ liveEditFailed: '`LiveEdit` failed: {PH1}', /** *@description Error text displayed in the console when compiling a live-edited script fails. LiveEdit is *the name of the feature for editing code that is already running. *@example {connection lost} PH1 */ liveEditCompileFailed: '`LiveEdit` compile failed: {PH1}', }; const str_ = i18n.i18n.registerUIStrings('models/bindings/ResourceScriptMapping.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); export class ResourceScriptMapping { debuggerModel; #workspace; debuggerWorkspaceBinding; #uiSourceCodeToScriptFile; #projects; #scriptToUISourceCode; #eventListeners; constructor(debuggerModel, workspace, debuggerWorkspaceBinding) { this.debuggerModel = debuggerModel; this.#workspace = workspace; this.debuggerWorkspaceBinding = debuggerWorkspaceBinding; this.#uiSourceCodeToScriptFile = new Map(); this.#projects = new Map(); this.#scriptToUISourceCode = new Map(); const runtimeModel = debuggerModel.runtimeModel(); this.#eventListeners = [ this.debuggerModel.addEventListener(SDK.DebuggerModel.Events.ParsedScriptSource, event => this.addScript(event.data), this), this.debuggerModel.addEventListener(SDK.DebuggerModel.Events.GlobalObjectCleared, this.globalObjectCleared, this), runtimeModel.addEventListener(SDK.RuntimeModel.Events.ExecutionContextDestroyed, this.executionContextDestroyed, this), runtimeModel.target().targetManager().addEventListener(SDK.TargetManager.Events.InspectedURLChanged, this.inspectedURLChanged, this), ]; } project(script) { const prefix = script.isContentScript() ? 'js:extensions:' : 'js::'; const projectId = prefix + this.debuggerModel.target().id() + ':' + script.frameId; let project = this.#projects.get(projectId); if (!project) { const projectType = script.isContentScript() ? Workspace.Workspace.projectTypes.ContentScripts : Workspace.Workspace.projectTypes.Network; project = new ContentProviderBasedProject(this.#workspace, projectId, projectType, '' /* displayName */, false /* isServiceProject */); NetworkProject.setTargetForProject(project, this.debuggerModel.target()); this.#projects.set(projectId, project); } return project; } uiSourceCodeForScript(script) { return this.#scriptToUISourceCode.get(script) ?? null; } rawLocationToUILocation(rawLocation) { const script = rawLocation.script(); if (!script) { return null; } const uiSourceCode = this.#scriptToUISourceCode.get(script); if (!uiSourceCode) { return null; } const scriptFile = this.#uiSourceCodeToScriptFile.get(uiSourceCode); if (!scriptFile) { return null; } if ((scriptFile.hasDivergedFromVM() && !scriptFile.isMergingToVM()) || scriptFile.isDivergingFromVM()) { return null; } if (scriptFile.script !== script) { return null; } const { lineNumber, columnNumber = 0 } = rawLocation; return uiSourceCode.uiLocation(lineNumber, columnNumber); } uiLocationToRawLocations(uiSourceCode, lineNumber, columnNumber) { const scriptFile = this.#uiSourceCodeToScriptFile.get(uiSourceCode); if (!scriptFile) { return []; } const { script } = scriptFile; if (!script) { return []; } return [this.debuggerModel.createRawLocation(script, lineNumber, columnNumber)]; } uiLocationRangeToRawLocationRanges(uiSourceCode, { startLine, startColumn, endLine, endColumn }) { const scriptFile = this.#uiSourceCodeToScriptFile.get(uiSourceCode); if (!scriptFile) { return null; } const { script } = scriptFile; if (!script) { return null; } const start = this.debuggerModel.createRawLocation(script, startLine, startColumn); const end = this.debuggerModel.createRawLocation(script, endLine, endColumn); return [{ start, end }]; } inspectedURLChanged(event) { for (let target = this.debuggerModel.target(); target !== event.data; target = target.parentTarget()) { if (target === null) { return; } } // Just remove and readd all scripts to ensure their URLs are reflected correctly. for (const script of Array.from(this.#scriptToUISourceCode.keys())) { this.removeScripts([script]); this.addScript(script); } } addScript(script) { // Ignore live edit scripts here. if (script.isLiveEdit() || script.isBreakpointCondition) { return; } let url = script.sourceURL; if (!url) { return; } if (script.hasSourceURL) { // Try to resolve `//# sourceURL=` annotations relative to // the base URL, according to the sourcemap specification. url = SDK.SourceMapManager.SourceMapManager.resolveRelativeSourceURL(script.debuggerModel.target(), url); } else { // Ignore inline <script>s without `//# sourceURL` annotation here. if (script.isInlineScript()) { return; } // Filter out embedder injected content scripts. if (script.isContentScript()) { const parsedURL = new Common.ParsedURL.ParsedURL(url); if (!parsedURL.isValid) { return; } } } // Remove previous UISourceCode, if any const project = this.project(script); const oldUISourceCode = project.uiSourceCodeForURL(url); if (oldUISourceCode) { const oldScriptFile = this.#uiSourceCodeToScriptFile.get(oldUISourceCode); if (oldScriptFile && oldScriptFile.script) { this.removeScripts([oldScriptFile.script]); } } // Create UISourceCode. const originalContentProvider = script.originalContentProvider(); const uiSourceCode = project.createUISourceCode(url, originalContentProvider.contentType()); NetworkProject.setInitialFrameAttribution(uiSourceCode, script.frameId); const metadata = metadataForURL(this.debuggerModel.target(), script.frameId, url); // Bind UISourceCode to scripts. const scriptFile = new ResourceScriptFile(this, uiSourceCode, script); this.#uiSourceCodeToScriptFile.set(uiSourceCode, scriptFile); this.#scriptToUISourceCode.set(script, uiSourceCode); const mimeType = script.isWasm() ? 'application/wasm' : 'text/javascript'; project.addUISourceCodeWithProvider(uiSourceCode, originalContentProvider, metadata, mimeType); void this.debuggerWorkspaceBinding.updateLocations(script); } scriptFile(uiSourceCode) { return this.#uiSourceCodeToScriptFile.get(uiSourceCode) || null; } removeScripts(scripts) { const uiSourceCodesByProject = new Platform.MapUtilities.Multimap(); for (const script of scripts) { const uiSourceCode = this.#scriptToUISourceCode.get(script); if (!uiSourceCode) { continue; } const scriptFile = this.#uiSourceCodeToScriptFile.get(uiSourceCode); if (scriptFile) { scriptFile.dispose(); } this.#uiSourceCodeToScriptFile.delete(uiSourceCode); this.#scriptToUISourceCode.delete(script); uiSourceCodesByProject.set(uiSourceCode.project(), uiSourceCode); void this.debuggerWorkspaceBinding.updateLocations(script); } for (const project of uiSourceCodesByProject.keysArray()) { const uiSourceCodes = uiSourceCodesByProject.get(project); // Check if all the ui source codes in the project are in |uiSourceCodes|. let allInProjectRemoved = true; for (const projectSourceCode of project.uiSourceCodes()) { if (!uiSourceCodes.has(projectSourceCode)) { allInProjectRemoved = false; break; } } // Drop the whole project if no source codes are left in it. if (allInProjectRemoved) { this.#projects.delete(project.id()); project.removeProject(); } else { // Otherwise, announce the removal of each UI source code individually. uiSourceCodes.forEach(c => project.removeUISourceCode(c.url())); } } } executionContextDestroyed(event) { const executionContext = event.data; this.removeScripts(this.debuggerModel.scriptsForExecutionContext(executionContext)); } globalObjectCleared() { const scripts = Array.from(this.#scriptToUISourceCode.keys()); this.removeScripts(scripts); } resetForTest() { this.globalObjectCleared(); } dispose() { Common.EventTarget.removeEventListeners(this.#eventListeners); this.globalObjectCleared(); } } export class ResourceScriptFile extends Common.ObjectWrapper.ObjectWrapper { #resourceScriptMapping; #uiSourceCodeInternal; #script; #scriptSource; #isDivergingFromVMInternal; #hasDivergedFromVMInternal; #isMergingToVMInternal; #updateMutex = new Common.Mutex.Mutex(); constructor(resourceScriptMapping, uiSourceCode, script) { super(); this.#resourceScriptMapping = resourceScriptMapping; this.#uiSourceCodeInternal = uiSourceCode; if (this.#uiSourceCodeInternal.contentType().isScript()) { this.#script = script; } this.#uiSourceCodeInternal.addEventListener(Workspace.UISourceCode.Events.WorkingCopyChanged, this.workingCopyChanged, this); this.#uiSourceCodeInternal.addEventListener(Workspace.UISourceCode.Events.WorkingCopyCommitted, this.workingCopyCommitted, this); } isDiverged() { if (this.#uiSourceCodeInternal.isDirty()) { return true; } if (!this.#script) { return false; } if (typeof this.#scriptSource === 'undefined' || this.#scriptSource === null) { return false; } const workingCopy = this.#uiSourceCodeInternal.workingCopy(); if (!workingCopy) { return false; } // Match ignoring sourceURL. if (!workingCopy.startsWith(this.#scriptSource.trimEnd())) { return true; } const suffix = this.#uiSourceCodeInternal.workingCopy().substr(this.#scriptSource.length); return Boolean(suffix.length) && !suffix.match(SDK.Script.sourceURLRegex); } workingCopyChanged() { void this.update(); } workingCopyCommitted() { if (this.#uiSourceCodeInternal.project().canSetFileContent()) { return; } if (!this.#script) { return; } const source = this.#uiSourceCodeInternal.workingCopy(); void this.#script.editSource(source).then(({ status, exceptionDetails }) => { void this.scriptSourceWasSet(source, status, exceptionDetails); }); } async scriptSourceWasSet(source, status, exceptionDetails) { if (status === "Ok" /* Protocol.Debugger.SetScriptSourceResponseStatus.Ok */) { this.#scriptSource = source; } await this.update(); if (status === "Ok" /* Protocol.Debugger.SetScriptSourceResponseStatus.Ok */) { return; } if (!exceptionDetails) { // TODO(crbug.com/1334484): Instead of to the console, report these errors in an "info bar" at the bottom // of the text editor, similar to e.g. source mapping errors. Common.Console.Console.instance().addMessage(i18nString(UIStrings.liveEditFailed, { PH1: getErrorText(status) }), Common.Console.MessageLevel.Warning); return; } const messageText = i18nString(UIStrings.liveEditCompileFailed, { PH1: exceptionDetails.text }); this.#uiSourceCodeInternal.addLineMessage(Workspace.UISourceCode.Message.Level.Error, messageText, exceptionDetails.lineNumber, exceptionDetails.columnNumber); function getErrorText(status) { switch (status) { case "BlockedByActiveFunction" /* Protocol.Debugger.SetScriptSourceResponseStatus.BlockedByActiveFunction */: return 'Functions that are on the stack (currently being executed) can not be edited'; case "BlockedByActiveGenerator" /* Protocol.Debugger.SetScriptSourceResponseStatus.BlockedByActiveGenerator */: return 'Async functions/generators that are active can not be edited'; case "BlockedByTopLevelEsModuleChange" /* Protocol.Debugger.SetScriptSourceResponseStatus.BlockedByTopLevelEsModuleChange */: return 'The top-level of ES modules can not be edited'; case "CompileError" /* Protocol.Debugger.SetScriptSourceResponseStatus.CompileError */: case "Ok" /* Protocol.Debugger.SetScriptSourceResponseStatus.Ok */: throw new Error('Compile errors and Ok status must not be reported on the console'); } } } async update() { // Do not interleave "divergeFromVM" with "mergeToVM" calls. const release = await this.#updateMutex.acquire(); if (this.isDiverged() && !this.#hasDivergedFromVMInternal) { await this.divergeFromVM(); } else if (!this.isDiverged() && this.#hasDivergedFromVMInternal) { await this.mergeToVM(); } release(); } async divergeFromVM() { if (this.#script) { this.#isDivergingFromVMInternal = true; await this.#resourceScriptMapping.debuggerWorkspaceBinding.updateLocations(this.#script); this.#isDivergingFromVMInternal = undefined; this.#hasDivergedFromVMInternal = true; this.dispatchEventToListeners("DidDivergeFromVM" /* ResourceScriptFile.Events.DidDivergeFromVM */); } } async mergeToVM() { if (this.#script) { this.#hasDivergedFromVMInternal = undefined; this.#isMergingToVMInternal = true; await this.#resourceScriptMapping.debuggerWorkspaceBinding.updateLocations(this.#script); this.#isMergingToVMInternal = undefined; this.dispatchEventToListeners("DidMergeToVM" /* ResourceScriptFile.Events.DidMergeToVM */); } } hasDivergedFromVM() { return Boolean(this.#hasDivergedFromVMInternal); } isDivergingFromVM() { return Boolean(this.#isDivergingFromVMInternal); } isMergingToVM() { return Boolean(this.#isMergingToVMInternal); } checkMapping() { if (!this.#script || typeof this.#scriptSource !== 'undefined') { this.mappingCheckedForTest(); return; } void this.#script.requestContent().then(deferredContent => { this.#scriptSource = deferredContent.content; void this.update().then(() => this.mappingCheckedForTest()); }); } mappingCheckedForTest() { } dispose() { this.#uiSourceCodeInternal.removeEventListener(Workspace.UISourceCode.Events.WorkingCopyChanged, this.workingCopyChanged, this); this.#uiSourceCodeInternal.removeEventListener(Workspace.UISourceCode.Events.WorkingCopyCommitted, this.workingCopyCommitted, this); } addSourceMapURL(sourceMapURL) { if (!this.#script) { return; } this.#script.debuggerModel.setSourceMapURL(this.#script, sourceMapURL); } addDebugInfoURL(debugInfoURL) { if (!this.#script) { return; } const { pluginManager } = DebuggerWorkspaceBinding.instance(); if (pluginManager) { pluginManager.setDebugInfoURL(this.#script, debugInfoURL); } } hasSourceMapURL() { return this.#script !== undefined && Boolean(this.#script.sourceMapURL); } async missingSymbolFiles() { if (!this.#script) { return null; } const { pluginManager } = this.#resourceScriptMapping.debuggerWorkspaceBinding; if (!pluginManager) { return null; } const sources = await pluginManager.getSourcesForScript(this.#script); return sources && 'missingSymbolFiles' in sources ? sources.missingSymbolFiles : null; } get script() { return this.#script || null; } get uiSourceCode() { return this.#uiSourceCodeInternal; } } //# sourceMappingURL=ResourceScriptMapping.js.map