| /* |
| * Copyright (C) 2011 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. |
| */ |
| |
| export default class ResourceTreeModel extends SDK.SDKModel { |
| /** |
| * @param {!SDK.Target} target |
| */ |
| constructor(target) { |
| super(target); |
| |
| const networkManager = target.model(SDK.NetworkManager); |
| if (networkManager) { |
| networkManager.addEventListener(SDK.NetworkManager.Events.RequestFinished, this._onRequestFinished, this); |
| networkManager.addEventListener( |
| SDK.NetworkManager.Events.RequestUpdateDropped, this._onRequestUpdateDropped, this); |
| } |
| this._agent = target.pageAgent(); |
| this._agent.enable(); |
| this._securityOriginManager = target.model(SDK.SecurityOriginManager); |
| |
| target.registerPageDispatcher(new SDK.PageDispatcher(this)); |
| |
| /** @type {!Map<string, !ResourceTreeFrame>} */ |
| this._frames = new Map(); |
| this._cachedResourcesProcessed = false; |
| this._pendingReloadOptions = null; |
| this._reloadSuspensionCount = 0; |
| this._isInterstitialShowing = false; |
| /** @type {?ResourceTreeFrame} */ |
| this.mainFrame = null; |
| |
| this._agent.getResourceTree().then(this._processCachedResources.bind(this)); |
| } |
| |
| /** |
| * @param {!SDK.NetworkRequest} request |
| * @return {?ResourceTreeFrame} |
| */ |
| static frameForRequest(request) { |
| const networkManager = SDK.NetworkManager.forRequest(request); |
| const resourceTreeModel = networkManager ? networkManager.target().model(ResourceTreeModel) : null; |
| if (!resourceTreeModel) { |
| return null; |
| } |
| return resourceTreeModel.frameForId(request.frameId); |
| } |
| |
| /** |
| * @return {!Array.<!ResourceTreeFrame>} |
| */ |
| static frames() { |
| let result = []; |
| for (const resourceTreeModel of SDK.targetManager.models(ResourceTreeModel)) { |
| result = result.concat(resourceTreeModel._frames.valuesArray()); |
| } |
| return result; |
| } |
| |
| /** |
| * @param {string} url |
| * @return {?SDK.Resource} |
| */ |
| static resourceForURL(url) { |
| for (const resourceTreeModel of SDK.targetManager.models(ResourceTreeModel)) { |
| const mainFrame = resourceTreeModel.mainFrame; |
| const result = mainFrame ? mainFrame.resourceForURL(url) : null; |
| if (result) { |
| return result; |
| } |
| } |
| return null; |
| } |
| |
| /** |
| * @param {boolean=} bypassCache |
| * @param {string=} scriptToEvaluateOnLoad |
| */ |
| static reloadAllPages(bypassCache, scriptToEvaluateOnLoad) { |
| for (const resourceTreeModel of SDK.targetManager.models(ResourceTreeModel)) { |
| if (!resourceTreeModel.target().parentTarget()) { |
| resourceTreeModel.reloadPage(bypassCache, scriptToEvaluateOnLoad); |
| } |
| } |
| } |
| |
| /** |
| * @return {!SDK.DOMModel} |
| */ |
| domModel() { |
| return /** @type {!SDK.DOMModel} */ (this.target().model(SDK.DOMModel)); |
| } |
| |
| /** |
| * @param {?Protocol.Page.FrameResourceTree} mainFramePayload |
| */ |
| _processCachedResources(mainFramePayload) { |
| if (mainFramePayload) { |
| this.dispatchEventToListeners(Events.WillLoadCachedResources); |
| this._addFramesRecursively(null, mainFramePayload); |
| this.target().setInspectedURL(mainFramePayload.frame.url); |
| } |
| this._cachedResourcesProcessed = true; |
| const runtimeModel = this.target().model(SDK.RuntimeModel); |
| if (runtimeModel) { |
| runtimeModel.setExecutionContextComparator(this._executionContextComparator.bind(this)); |
| runtimeModel.fireExecutionContextOrderChanged(); |
| } |
| this.dispatchEventToListeners(Events.CachedResourcesLoaded, this); |
| } |
| |
| /** |
| * @return {boolean} |
| */ |
| cachedResourcesLoaded() { |
| return this._cachedResourcesProcessed; |
| } |
| |
| /** |
| * @return {boolean} |
| */ |
| isInterstitialShowing() { |
| return this._isInterstitialShowing; |
| } |
| |
| /** |
| * @param {!ResourceTreeFrame} frame |
| * @param {boolean=} aboutToNavigate |
| */ |
| _addFrame(frame, aboutToNavigate) { |
| this._frames.set(frame.id, frame); |
| if (frame.isMainFrame()) { |
| this.mainFrame = frame; |
| } |
| this.dispatchEventToListeners(Events.FrameAdded, frame); |
| this._updateSecurityOrigins(); |
| } |
| |
| /** |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {?Protocol.Page.FrameId} parentFrameId |
| * @param {!Protocol.Runtime.StackTrace=} stackTrace |
| * @return {?ResourceTreeFrame} |
| */ |
| _frameAttached(frameId, parentFrameId, stackTrace) { |
| const parentFrame = parentFrameId ? (this._frames.get(parentFrameId) || null) : null; |
| // Do nothing unless cached resource tree is processed - it will overwrite everything. |
| if (!this._cachedResourcesProcessed && parentFrame) { |
| return null; |
| } |
| if (this._frames.has(frameId)) { |
| return null; |
| } |
| |
| const frame = new ResourceTreeFrame(this, parentFrame, frameId, null, stackTrace || null); |
| if (parentFrameId && !parentFrame) { |
| frame._crossTargetParentFrameId = parentFrameId; |
| } |
| if (frame.isMainFrame() && this.mainFrame) { |
| // Navigation to the new backend process. |
| this._frameDetached(this.mainFrame.id); |
| } |
| this._addFrame(frame, true); |
| return frame; |
| } |
| |
| /** |
| * @param {!Protocol.Page.Frame} framePayload |
| */ |
| _frameNavigated(framePayload) { |
| const parentFrame = framePayload.parentId ? (this._frames.get(framePayload.parentId) || null) : null; |
| // Do nothing unless cached resource tree is processed - it will overwrite everything. |
| if (!this._cachedResourcesProcessed && parentFrame) { |
| return; |
| } |
| let frame = this._frames.get(framePayload.id); |
| if (!frame) { |
| // Simulate missed "frameAttached" for a main frame navigation to the new backend process. |
| frame = this._frameAttached(framePayload.id, framePayload.parentId || ''); |
| console.assert(frame); |
| } |
| |
| this.dispatchEventToListeners(Events.FrameWillNavigate, frame); |
| frame._navigate(framePayload); |
| this.dispatchEventToListeners(Events.FrameNavigated, frame); |
| |
| if (frame.isMainFrame()) { |
| this.dispatchEventToListeners(Events.MainFrameNavigated, frame); |
| } |
| |
| // Fill frame with retained resources (the ones loaded using new loader). |
| const resources = frame.resources(); |
| for (let i = 0; i < resources.length; ++i) { |
| this.dispatchEventToListeners(Events.ResourceAdded, resources[i]); |
| } |
| |
| if (frame.isMainFrame()) { |
| this.target().setInspectedURL(frame.url); |
| } |
| this._updateSecurityOrigins(); |
| } |
| |
| /** |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| _frameDetached(frameId) { |
| // Do nothing unless cached resource tree is processed - it will overwrite everything. |
| if (!this._cachedResourcesProcessed) { |
| return; |
| } |
| |
| const frame = this._frames.get(frameId); |
| if (!frame) { |
| return; |
| } |
| |
| if (frame.parentFrame) { |
| frame.parentFrame._removeChildFrame(frame); |
| } else { |
| frame._remove(); |
| } |
| this._updateSecurityOrigins(); |
| } |
| |
| /** |
| * @param {!Common.Event} event |
| */ |
| _onRequestFinished(event) { |
| if (!this._cachedResourcesProcessed) { |
| return; |
| } |
| |
| const request = /** @type {!SDK.NetworkRequest} */ (event.data); |
| if (request.failed || request.resourceType() === Common.resourceTypes.XHR) { |
| return; |
| } |
| |
| const frame = this._frames.get(request.frameId); |
| if (frame) { |
| frame._addRequest(request); |
| } |
| } |
| |
| /** |
| * @param {!Common.Event} event |
| */ |
| _onRequestUpdateDropped(event) { |
| if (!this._cachedResourcesProcessed) { |
| return; |
| } |
| |
| const frameId = event.data.frameId; |
| const frame = this._frames.get(frameId); |
| if (!frame) { |
| return; |
| } |
| |
| const url = event.data.url; |
| if (frame._resourcesMap[url]) { |
| return; |
| } |
| |
| const resource = new SDK.Resource( |
| this, null, url, frame.url, frameId, event.data.loaderId, Common.resourceTypes[event.data.resourceType], |
| event.data.mimeType, event.data.lastModified, null); |
| frame.addResource(resource); |
| } |
| |
| /** |
| * @param {!Protocol.Page.FrameId} frameId |
| * @return {!ResourceTreeFrame} |
| */ |
| frameForId(frameId) { |
| return this._frames.get(frameId); |
| } |
| |
| /** |
| * @param {function(!SDK.Resource)} callback |
| * @return {boolean} |
| */ |
| forAllResources(callback) { |
| if (this.mainFrame) { |
| return this.mainFrame._callForFrameResources(callback); |
| } |
| return false; |
| } |
| |
| /** |
| * @return {!Array<!ResourceTreeFrame>} |
| */ |
| frames() { |
| return this._frames.valuesArray(); |
| } |
| |
| /** |
| * @param {string} url |
| * @return {?SDK.Resource} |
| */ |
| resourceForURL(url) { |
| // Workers call into this with no frames available. |
| return this.mainFrame ? this.mainFrame.resourceForURL(url) : null; |
| } |
| |
| /** |
| * @param {?ResourceTreeFrame} parentFrame |
| * @param {!Protocol.Page.FrameResourceTree} frameTreePayload |
| */ |
| _addFramesRecursively(parentFrame, frameTreePayload) { |
| const framePayload = frameTreePayload.frame; |
| const frame = new ResourceTreeFrame(this, parentFrame, framePayload.id, framePayload, null); |
| if (!parentFrame && framePayload.parentId) { |
| frame._crossTargetParentFrameId = framePayload.parentId; |
| } |
| this._addFrame(frame); |
| |
| for (let i = 0; frameTreePayload.childFrames && i < frameTreePayload.childFrames.length; ++i) { |
| this._addFramesRecursively(frame, frameTreePayload.childFrames[i]); |
| } |
| |
| for (let i = 0; i < frameTreePayload.resources.length; ++i) { |
| const subresource = frameTreePayload.resources[i]; |
| const resource = this._createResourceFromFramePayload( |
| framePayload, subresource.url, Common.resourceTypes[subresource.type], subresource.mimeType, |
| subresource.lastModified || null, subresource.contentSize || null); |
| frame.addResource(resource); |
| } |
| |
| if (!frame._resourcesMap[framePayload.url]) { |
| const frameResource = this._createResourceFromFramePayload( |
| framePayload, framePayload.url, Common.resourceTypes.Document, framePayload.mimeType, null, null); |
| frame.addResource(frameResource); |
| } |
| } |
| |
| /** |
| * @param {!Protocol.Page.Frame} frame |
| * @param {string} url |
| * @param {!Common.ResourceType} type |
| * @param {string} mimeType |
| * @param {?number} lastModifiedTime |
| * @param {?number} contentSize |
| * @return {!SDK.Resource} |
| */ |
| _createResourceFromFramePayload(frame, url, type, mimeType, lastModifiedTime, contentSize) { |
| const lastModified = typeof lastModifiedTime === 'number' ? new Date(lastModifiedTime * 1000) : null; |
| return new SDK.Resource( |
| this, null, url, frame.url, frame.id, frame.loaderId, type, mimeType, lastModified, contentSize); |
| } |
| |
| suspendReload() { |
| this._reloadSuspensionCount++; |
| } |
| |
| resumeReload() { |
| this._reloadSuspensionCount--; |
| console.assert(this._reloadSuspensionCount >= 0, 'Unbalanced call to ResourceTreeModel.resumeReload()'); |
| if (!this._reloadSuspensionCount && this._pendingReloadOptions) { |
| this.reloadPage.apply(this, this._pendingReloadOptions); |
| } |
| } |
| |
| /** |
| * @param {boolean=} bypassCache |
| * @param {string=} scriptToEvaluateOnLoad |
| */ |
| reloadPage(bypassCache, scriptToEvaluateOnLoad) { |
| // Only dispatch PageReloadRequested upon first reload request to simplify client logic. |
| if (!this._pendingReloadOptions) { |
| this.dispatchEventToListeners(Events.PageReloadRequested, this); |
| } |
| if (this._reloadSuspensionCount) { |
| this._pendingReloadOptions = [bypassCache, scriptToEvaluateOnLoad]; |
| return; |
| } |
| this._pendingReloadOptions = null; |
| this.dispatchEventToListeners(Events.WillReloadPage); |
| this._agent.reload(bypassCache, scriptToEvaluateOnLoad); |
| } |
| |
| /** |
| * @param {string} url |
| * @return {!Promise} |
| */ |
| navigate(url) { |
| return this._agent.navigate(url); |
| } |
| |
| /** |
| * @return {!Promise<?{currentIndex: number, entries: !Array<!Protocol.Page.NavigationEntry>}>} |
| */ |
| async navigationHistory() { |
| const response = await this._agent.invoke_getNavigationHistory({}); |
| if (response[Protocol.Error]) { |
| return null; |
| } |
| return {currentIndex: response.currentIndex, entries: response.entries}; |
| } |
| |
| /** |
| * @param {!Protocol.Page.NavigationEntry} entry |
| */ |
| navigateToHistoryEntry(entry) { |
| this._agent.navigateToHistoryEntry(entry.id); |
| } |
| |
| /** |
| * @return {!Promise<{url: string, data: ?string, errors: !Array<!Protocol.Page.AppManifestError>}>} |
| */ |
| async fetchAppManifest() { |
| const response = await this._agent.invoke_getAppManifest({}); |
| if (response[Protocol.Error]) { |
| return {url: response.url, data: null, errors: []}; |
| } |
| return {url: response.url, data: response.data || null, errors: response.errors}; |
| } |
| |
| /** |
| * @return {!Promise<!Array<string>>} |
| */ |
| async getInstallabilityErrors() { |
| const response = await this._agent.invoke_getInstallabilityErrors({}); |
| return response.errors || []; |
| } |
| |
| /** |
| * @param {!SDK.ExecutionContext} a |
| * @param {!SDK.ExecutionContext} b |
| * @return {number} |
| */ |
| _executionContextComparator(a, b) { |
| /** |
| * @param {!ResourceTreeFrame} frame |
| * @return {!Array<!ResourceTreeFrame>} |
| */ |
| function framePath(frame) { |
| let currentFrame = frame; |
| const parents = []; |
| while (currentFrame) { |
| parents.push(currentFrame); |
| currentFrame = currentFrame.parentFrame; |
| } |
| return parents.reverse(); |
| } |
| |
| if (a.target() !== b.target()) { |
| return SDK.ExecutionContext.comparator(a, b); |
| } |
| |
| const framesA = a.frameId ? framePath(this.frameForId(a.frameId)) : []; |
| const framesB = b.frameId ? framePath(this.frameForId(b.frameId)) : []; |
| let frameA; |
| let frameB; |
| for (let i = 0;; i++) { |
| if (!framesA[i] || !framesB[i] || (framesA[i] !== framesB[i])) { |
| frameA = framesA[i]; |
| frameB = framesB[i]; |
| break; |
| } |
| } |
| if (!frameA && frameB) { |
| return -1; |
| } |
| |
| if (!frameB && frameA) { |
| return 1; |
| } |
| |
| if (frameA && frameB) { |
| return frameA.id.localeCompare(frameB.id); |
| } |
| |
| return SDK.ExecutionContext.comparator(a, b); |
| } |
| |
| /** |
| * @return {!SDK.ResourceTreeModel.SecurityOriginData} |
| */ |
| _getSecurityOriginData() { |
| /** @type {!Set<string>} */ |
| const securityOrigins = new Set(); |
| |
| let mainSecurityOrigin = null; |
| let unreachableMainSecurityOrigin = null; |
| for (const frame of this._frames.values()) { |
| const origin = frame.securityOrigin; |
| if (!origin) { |
| continue; |
| } |
| |
| securityOrigins.add(origin); |
| if (frame.isMainFrame()) { |
| mainSecurityOrigin = origin; |
| if (frame.unreachableUrl()) { |
| const unreachableParsed = new Common.ParsedURL(frame.unreachableUrl()); |
| unreachableMainSecurityOrigin = unreachableParsed.securityOrigin(); |
| } |
| } |
| } |
| return { |
| securityOrigins: securityOrigins, |
| mainSecurityOrigin: mainSecurityOrigin, |
| unreachableMainSecurityOrigin: unreachableMainSecurityOrigin |
| }; |
| } |
| |
| _updateSecurityOrigins() { |
| const data = this._getSecurityOriginData(); |
| this._securityOriginManager.setMainSecurityOrigin( |
| data.mainSecurityOrigin || '', data.unreachableMainSecurityOrigin || ''); |
| this._securityOriginManager.updateSecurityOrigins(data.securityOrigins); |
| } |
| |
| /** |
| * @return {?string} |
| */ |
| getMainSecurityOrigin() { |
| const data = this._getSecurityOriginData(); |
| return data.mainSecurityOrigin || data.unreachableMainSecurityOrigin; |
| } |
| } |
| |
| /** @enum {symbol} */ |
| export const Events = { |
| FrameAdded: Symbol('FrameAdded'), |
| FrameNavigated: Symbol('FrameNavigated'), |
| FrameDetached: Symbol('FrameDetached'), |
| FrameResized: Symbol('FrameResized'), |
| FrameWillNavigate: Symbol('FrameWillNavigate'), |
| MainFrameNavigated: Symbol('MainFrameNavigated'), |
| ResourceAdded: Symbol('ResourceAdded'), |
| WillLoadCachedResources: Symbol('WillLoadCachedResources'), |
| CachedResourcesLoaded: Symbol('CachedResourcesLoaded'), |
| DOMContentLoaded: Symbol('DOMContentLoaded'), |
| LifecycleEvent: Symbol('LifecycleEvent'), |
| Load: Symbol('Load'), |
| PageReloadRequested: Symbol('PageReloadRequested'), |
| WillReloadPage: Symbol('WillReloadPage'), |
| InterstitialShown: Symbol('InterstitialShown'), |
| InterstitialHidden: Symbol('InterstitialHidden') |
| }; |
| |
| /** |
| * @unrestricted |
| */ |
| export class ResourceTreeFrame { |
| /** |
| * @param {!ResourceTreeModel} model |
| * @param {?ResourceTreeFrame} parentFrame |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {?Protocol.Page.Frame} payload |
| * @param {?Protocol.Runtime.StackTrace} creationStackTrace |
| */ |
| constructor(model, parentFrame, frameId, payload, creationStackTrace) { |
| this._model = model; |
| this._parentFrame = parentFrame; |
| this._id = frameId; |
| this._url = ''; |
| this._crossTargetParentFrameId = null; |
| |
| if (payload) { |
| this._loaderId = payload.loaderId; |
| this._name = payload.name; |
| this._url = payload.url; |
| this._securityOrigin = payload.securityOrigin; |
| this._mimeType = payload.mimeType; |
| this._unreachableUrl = payload.unreachableUrl || ''; |
| } |
| |
| this._creationStackTrace = creationStackTrace; |
| |
| /** |
| * @type {!Array.<!ResourceTreeFrame>} |
| */ |
| this._childFrames = []; |
| |
| /** |
| * @type {!Object.<string, !SDK.Resource>} |
| */ |
| this._resourcesMap = {}; |
| |
| if (this._parentFrame) { |
| this._parentFrame._childFrames.push(this); |
| } |
| } |
| |
| |
| /** |
| * @param {!Protocol.Page.Frame} framePayload |
| */ |
| _navigate(framePayload) { |
| this._loaderId = framePayload.loaderId; |
| this._name = framePayload.name; |
| this._url = framePayload.url; |
| this._securityOrigin = framePayload.securityOrigin; |
| this._mimeType = framePayload.mimeType; |
| this._unreachableUrl = framePayload.unreachableUrl || ''; |
| const mainResource = this._resourcesMap[this._url]; |
| this._resourcesMap = {}; |
| this._removeChildFrames(); |
| if (mainResource && mainResource.loaderId === this._loaderId) { |
| this.addResource(mainResource); |
| } |
| } |
| |
| /** |
| * @return {!ResourceTreeModel} |
| */ |
| resourceTreeModel() { |
| return this._model; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| get id() { |
| return this._id; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| get name() { |
| return this._name || ''; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| get url() { |
| return this._url; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| get securityOrigin() { |
| return this._securityOrigin; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| unreachableUrl() { |
| return this._unreachableUrl; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| get loaderId() { |
| return this._loaderId; |
| } |
| |
| /** |
| * @return {?ResourceTreeFrame} |
| */ |
| get parentFrame() { |
| return this._parentFrame; |
| } |
| |
| /** |
| * @return {!Array.<!ResourceTreeFrame>} |
| */ |
| get childFrames() { |
| return this._childFrames; |
| } |
| |
| /** |
| * @return {?ResourceTreeFrame} |
| */ |
| crossTargetParentFrame() { |
| if (!this._crossTargetParentFrameId) { |
| return null; |
| } |
| if (!this._model.target().parentTarget()) { |
| return null; |
| } |
| const parentModel = this._model.target().parentTarget().model(ResourceTreeModel); |
| if (!parentModel) { |
| return null; |
| } |
| // Note that parent model has already processed cached resources: |
| // - when parent target was created, we issued getResourceTree call; |
| // - strictly after we issued setAutoAttach call; |
| // - both of them were handled in renderer in the same order; |
| // - cached resource tree got processed on parent model; |
| // - child target was created as a result of setAutoAttach call. |
| return parentModel._frames.get(this._crossTargetParentFrameId) || null; |
| } |
| |
| /** |
| * @param {function(!Protocol.Runtime.CallFrame):boolean} searchFn |
| * @return {?Protocol.Runtime.CallFrame} |
| */ |
| findCreationCallFrame(searchFn) { |
| let stackTrace = this._creationStackTrace; |
| while (stackTrace) { |
| const foundEntry = stackTrace.callFrames.find(searchFn); |
| if (foundEntry) { |
| return foundEntry; |
| } |
| stackTrace = this.parent; |
| } |
| return null; |
| } |
| |
| /** |
| * @return {boolean} |
| */ |
| isMainFrame() { |
| return !this._parentFrame; |
| } |
| |
| isTopFrame() { |
| return !this._parentFrame && !this._crossTargetParentFrameId; |
| } |
| |
| /** |
| * @return {!SDK.Resource} |
| */ |
| get mainResource() { |
| return this._resourcesMap[this._url]; |
| } |
| |
| /** |
| * @param {!ResourceTreeFrame} frame |
| */ |
| _removeChildFrame(frame) { |
| this._childFrames.remove(frame); |
| frame._remove(); |
| } |
| |
| _removeChildFrames() { |
| const frames = this._childFrames; |
| this._childFrames = []; |
| for (let i = 0; i < frames.length; ++i) { |
| frames[i]._remove(); |
| } |
| } |
| |
| _remove() { |
| this._removeChildFrames(); |
| this._model._frames.delete(this.id); |
| this._model.dispatchEventToListeners(Events.FrameDetached, this); |
| } |
| |
| /** |
| * @param {!SDK.Resource} resource |
| */ |
| addResource(resource) { |
| if (this._resourcesMap[resource.url] === resource) { |
| // Already in the tree, we just got an extra update. |
| return; |
| } |
| this._resourcesMap[resource.url] = resource; |
| this._model.dispatchEventToListeners(Events.ResourceAdded, resource); |
| } |
| |
| /** |
| * @param {!SDK.NetworkRequest} request |
| */ |
| _addRequest(request) { |
| let resource = this._resourcesMap[request.url()]; |
| if (resource && resource.request === request) { |
| // Already in the tree, we just got an extra update. |
| return; |
| } |
| resource = new SDK.Resource( |
| this._model, request, request.url(), request.documentURL, request.frameId, request.loaderId, |
| request.resourceType(), request.mimeType, null, null); |
| this._resourcesMap[resource.url] = resource; |
| this._model.dispatchEventToListeners(Events.ResourceAdded, resource); |
| } |
| |
| /** |
| * @return {!Array.<!SDK.Resource>} |
| */ |
| resources() { |
| const result = []; |
| for (const url in this._resourcesMap) { |
| result.push(this._resourcesMap[url]); |
| } |
| return result; |
| } |
| |
| /** |
| * @param {string} url |
| * @return {?SDK.Resource} |
| */ |
| resourceForURL(url) { |
| let resource = this._resourcesMap[url] || null; |
| if (resource) { |
| return resource; |
| } |
| for (let i = 0; !resource && i < this._childFrames.length; ++i) { |
| resource = this._childFrames[i].resourceForURL(url); |
| } |
| return resource; |
| } |
| |
| /** |
| * @param {function(!SDK.Resource)} callback |
| * @return {boolean} |
| */ |
| _callForFrameResources(callback) { |
| for (const url in this._resourcesMap) { |
| if (callback(this._resourcesMap[url])) { |
| return true; |
| } |
| } |
| |
| for (let i = 0; i < this._childFrames.length; ++i) { |
| if (this._childFrames[i]._callForFrameResources(callback)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| displayName() { |
| if (this.isTopFrame()) { |
| return Common.UIString('top'); |
| } |
| const subtitle = new Common.ParsedURL(this._url).displayName; |
| if (subtitle) { |
| if (!this._name) { |
| return subtitle; |
| } |
| return this._name + ' (' + subtitle + ')'; |
| } |
| return Common.UIString('<iframe>'); |
| } |
| } |
| |
| /** |
| * @implements {Protocol.PageDispatcher} |
| * @unrestricted |
| */ |
| export class PageDispatcher { |
| /** |
| * @param {!ResourceTreeModel} resourceTreeModel |
| */ |
| constructor(resourceTreeModel) { |
| this._resourceTreeModel = resourceTreeModel; |
| } |
| |
| /** |
| * @override |
| * @param {number} time |
| */ |
| domContentEventFired(time) { |
| this._resourceTreeModel.dispatchEventToListeners(Events.DOMContentLoaded, time); |
| } |
| |
| /** |
| * @override |
| * @param {number} time |
| */ |
| loadEventFired(time) { |
| this._resourceTreeModel.dispatchEventToListeners( |
| Events.Load, {resourceTreeModel: this._resourceTreeModel, loadTime: time}); |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {!Protocol.Network.LoaderId} loaderId |
| * @param {string} name |
| * @param {number} time |
| */ |
| lifecycleEvent(frameId, loaderId, name, time) { |
| this._resourceTreeModel.dispatchEventToListeners(Events.LifecycleEvent, {frameId, name}); |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {!Protocol.Page.FrameId} parentFrameId |
| * @param {!Protocol.Runtime.StackTrace=} stackTrace |
| */ |
| frameAttached(frameId, parentFrameId, stackTrace) { |
| this._resourceTreeModel._frameAttached(frameId, parentFrameId, stackTrace); |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.Frame} frame |
| */ |
| frameNavigated(frame) { |
| this._resourceTreeModel._frameNavigated(frame); |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| frameDetached(frameId) { |
| this._resourceTreeModel._frameDetached(frameId); |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| frameStartedLoading(frameId) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| frameStoppedLoading(frameId) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| frameRequestedNavigation(frameId) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {number} delay |
| */ |
| frameScheduledNavigation(frameId, delay) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| */ |
| frameClearedScheduledNavigation(frameId) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {string} url |
| */ |
| navigatedWithinDocument(frameId, url) { |
| } |
| |
| /** |
| * @override |
| */ |
| frameResized() { |
| this._resourceTreeModel.dispatchEventToListeners(Events.FrameResized, null); |
| } |
| |
| /** |
| * @override |
| * @param {string} url |
| * @param {string} message |
| * @param {string} dialogType |
| * @param {boolean} hasBrowserHandler |
| * @param {string=} prompt |
| */ |
| javascriptDialogOpening(url, message, dialogType, hasBrowserHandler, prompt) { |
| if (!hasBrowserHandler) { |
| this._resourceTreeModel._agent.handleJavaScriptDialog(false); |
| } |
| } |
| |
| /** |
| * @override |
| * @param {boolean} result |
| * @param {string} userInput |
| */ |
| javascriptDialogClosed(result, userInput) { |
| } |
| |
| /** |
| * @override |
| * @param {string} data |
| * @param {!Protocol.Page.ScreencastFrameMetadata} metadata |
| * @param {number} sessionId |
| */ |
| screencastFrame(data, metadata, sessionId) { |
| } |
| |
| /** |
| * @override |
| * @param {boolean} visible |
| */ |
| screencastVisibilityChanged(visible) { |
| } |
| |
| /** |
| * @override |
| */ |
| interstitialShown() { |
| this._resourceTreeModel._isInterstitialShowing = true; |
| this._resourceTreeModel.dispatchEventToListeners(Events.InterstitialShown); |
| } |
| |
| /** |
| * @override |
| */ |
| interstitialHidden() { |
| this._resourceTreeModel._isInterstitialShowing = false; |
| this._resourceTreeModel.dispatchEventToListeners(Events.InterstitialHidden); |
| } |
| |
| /** |
| * @override |
| * @param {string} url |
| * @param {string} windowName |
| * @param {!Array<string>} windowFeatures |
| * @param {boolean} userGesture |
| */ |
| windowOpen(url, windowName, windowFeatures, userGesture) { |
| } |
| |
| /** |
| * @override |
| * @param {string} url |
| * @param {string} data |
| */ |
| compilationCacheProduced(url, data) { |
| } |
| |
| /** |
| * @override |
| * @param {string} mode |
| */ |
| fileChooserOpened(mode) { |
| } |
| |
| /** |
| * @override |
| * @param {!Protocol.Page.FrameId} frameId |
| * @param {string} url |
| */ |
| downloadWillBegin(frameId, url) { |
| } |
| } |
| |
| /* Legacy exported object */ |
| self.SDK = self.SDK || {}; |
| |
| /* Legacy exported object */ |
| SDK = SDK || {}; |
| |
| /** @constructor */ |
| SDK.ResourceTreeModel = ResourceTreeModel; |
| |
| /** @enum {symbol} */ |
| SDK.ResourceTreeModel.Events = Events; |
| |
| /** @constructor */ |
| SDK.ResourceTreeFrame = ResourceTreeFrame; |
| |
| /** @constructor */ |
| SDK.PageDispatcher = PageDispatcher; |
| |
| /** |
| * @typedef {{ |
| * securityOrigins: !Set<string>, |
| * mainSecurityOrigin: ?string, |
| * unreachableMainSecurityOrigin: ?string |
| * }} |
| */ |
| SDK.ResourceTreeModel.SecurityOriginData; |
| |
| SDK.SDKModel.register(ResourceTreeModel, SDK.Target.Capability.DOM, true); |