All Downloads are FREE. Search and download functionalities are using the official Maven repository.

META-INF.dirigible.dev-tools.sdk.IssuesModel.js Maven / Gradle / Ivy

// Copyright 2019 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

import * as Common from '../common/common.js';  // eslint-disable-line no-unused-vars

import {CookieModel} from './CookieModel.js';
import {AggregatedIssue, Issue} from './Issue.js';
import {Events as NetworkManagerEvents, NetworkManager} from './NetworkManager.js';
import {NetworkRequest} from './NetworkRequest.js';  // eslint-disable-line no-unused-vars
import * as RelatedIssue from './RelatedIssue.js';
import {Events as ResourceTreeModelEvents, ResourceTreeFrame, ResourceTreeModel} from './ResourceTreeModel.js';  // eslint-disable-line no-unused-vars
import {Capability, SDKModel, Target} from './SDKModel.js';  // eslint-disable-line no-unused-vars


/**
 * This class generates issues in the front-end based on information provided by the network panel. In the long
 * term, we might move this reporting to the back-end, but the current COVID-19 policy requires us to tone down
 * back-end changes until we are back at normal release cycle.
 */
export class NetworkIssueDetector {
  /**
   * @param {!Target} target
   * @param {!IssuesModel} issuesModel
   */
  constructor(target, issuesModel) {
    this._issuesModel = issuesModel;
    this._networkManager = target.model(NetworkManager);
    if (this._networkManager) {
      this._networkManager.addEventListener(NetworkManagerEvents.RequestFinished, this._handleRequestFinished, this);
    }
    for (const request of self.SDK.networkLog.requests()) {
      this._handleRequestFinished({data: request});
    }
  }

  /**
   * @param {!{data:*}} event
   */
  _handleRequestFinished(event) {
    const request = /** @type {!NetworkRequest} */ (event.data);
    const blockedReason = getCoepBlockedReason(request);
    if (blockedReason) {
      const resources = {requests: [{requestId: request.requestId()}]};
      const code = `CrossOriginEmbedderPolicy::${this._toCamelCase(blockedReason)}`;
      this._issuesModel.issueAdded({code, resources});
    }

    /**
     * @param {!NetworkRequest} request
     * @return {?string}
     */
    function getCoepBlockedReason(request) {
      if (!request.wasBlocked()) {
        return null;
      }
      const blockedReason = request.blockedReason() || null;
      if (blockedReason === Protocol.Network.BlockedReason.CoepFrameResourceNeedsCoepHeader ||
          blockedReason === Protocol.Network.BlockedReason.CorpNotSameOriginAfterDefaultedToSameOriginByCoep ||
          blockedReason === Protocol.Network.BlockedReason.CoopSandboxedIframeCannotNavigateToCoopPage ||
          blockedReason === Protocol.Network.BlockedReason.CorpNotSameSite ||
          blockedReason === Protocol.Network.BlockedReason.CorpNotSameOrigin) {
        return blockedReason;
      }
      return null;
    }
  }

  detach() {
    if (this._networkManager) {
      this._networkManager.removeEventListener(NetworkManagerEvents.RequestFinished, this._handleRequestFinished, this);
    }
  }

  /**
   * @param {string} string
   * @return {string}
   */
  _toCamelCase(string) {
    const result = string.replace(/-\p{ASCII}/gu, match => match.substr(1).toUpperCase());
    return result.replace(/^./, match => match.toUpperCase());
  }
}


/**
 * @implements {Protocol.AuditsDispatcher}
 */
export class IssuesModel extends SDKModel {
  /**
   * @param {!Target} target
   */
  constructor(target) {
    super(target);
    this._enabled = false;
    /** @type {!Array} */
    this._issues = [];
    /** @type {!Map} */
    this._aggregatedIssuesByCode = new Map();
    this._cookiesModel = target.model(CookieModel);
    /** @type {*} */
    this._auditsAgent = null;
    this._hasSeenMainFrameNavigated = false;

    this._networkManager = target.model(NetworkManager);
    const resourceTreeModel = /** @type {?ResourceTreeModel} */ (target.model(ResourceTreeModel));
    if (resourceTreeModel) {
      resourceTreeModel.addEventListener(
        ResourceTreeModelEvents.MainFrameNavigated, this._onMainFrameNavigated, this);
    }
    this._networkIssueDetector = null;
    this.ensureEnabled();
  }

  /**
   * @param {!Common.EventTarget.EventTargetEvent} event
   */
  _onMainFrameNavigated(event) {
    const mainFrame = /** @type {!ResourceTreeFrame} */ (event.data);
    const keptIssues = [];
    for (const issue of this._issues) {
      if (issue.isAssociatedWithRequestId(mainFrame.loaderId)) {
        keptIssues.push(issue);
      } else {
        this._disconnectIssue(issue);
      }
    }
    this._issues = keptIssues;
    this._aggregatedIssuesByCode.clear();
    for (const issue of this._issues) {
      this._aggregateIssue(issue);
    }
    this._hasSeenMainFrameNavigated = true;
    this.dispatchEventToListeners(Events.FullUpdateRequired);
  }

  /**
   * The `IssuesModel` requires at least one `MainFrameNavigated` event. Receiving
   * one implies that we have all the information for accurate issues.
   *
   * @return {boolean}
   */
  reloadForAccurateInformationRequired() {
    return !this._hasSeenMainFrameNavigated;
  }

  ensureEnabled() {
    if (this._enabled) {
      return;
    }

    this._enabled = true;
    this.target().registerAuditsDispatcher(this);
    this._auditsAgent = this.target().auditsAgent();
    this._auditsAgent.enable();
    this._networkIssueDetector = new NetworkIssueDetector(this.target(), this);
  }

  /**
   * @param {!Issue} issue
   * @returns {!AggregatedIssue}
   */
  _aggregateIssue(issue) {
    if (!this._aggregatedIssuesByCode.has(issue.code())) {
      this._aggregatedIssuesByCode.set(issue.code(), new AggregatedIssue(issue.code()));
    }
    const aggregatedIssue = this._aggregatedIssuesByCode.get(issue.code());
    aggregatedIssue.addInstance(issue);
    return aggregatedIssue;
  }

  /**
   * @override
   * TODO(chromium:1063765): Strengthen types.
   * @param {*} inspectorIssue
   */
  issueAdded(inspectorIssue) {
    const issues = this._createIssuesFromProtocolIssue(inspectorIssue);
    this._issues.push(...issues);

    for (const issue of issues) {
      this._connectIssue(issue);
      const aggregatedIssue = this._aggregateIssue(issue);
      this.dispatchEventToListeners(Events.AggregatedIssueUpdated, aggregatedIssue);
    }
  }

  /**
   * Each issue reported by the backend can result in multiple {!Issue} instances.
   * Handlers are simple functions hard-coded into a map. If no handler is found for
   * a given Issue code, the default behavior creates one {!Issue} per incoming backend
   * issue.
   * TODO(chromium:1063765): Strengthen types.
   * @param {*} inspectorIssue} inspectorIssue
   * @return {!Array}
   */
  _createIssuesFromProtocolIssue(inspectorIssue) {
    const handler = issueCodeHandlers.get(inspectorIssue.code);
    if (handler) {
      // TODO(chromium:1063765): Pass the details object here, not the full inspector issue.
      return handler(this, inspectorIssue);
    }

    return [new Issue(inspectorIssue.code, inspectorIssue.resources)];
  }

  /**
   *
   * @param {!Issue} issue
   */
  _connectIssue(issue) {
    const resources = issue.resources();
    if (!resources) {
      return;
    }
    if (resources.requests) {
      for (const resourceRequest of resources.requests) {
        const request =
            /** @type {?NetworkRequest} */ (
                self.SDK.networkLog.requests().find(r => r.requestId() === resourceRequest.requestId));
        if (request) {
          // Connect the real network request with this issue and vice versa.
          RelatedIssue.connect(request, issue.getCategory(), issue);
          resourceRequest.request = request;
        }
      }
    }
  }

  /**
   *
   * @param {!Issue} issue
   */
  _disconnectIssue(issue) {
    const resources = issue.resources();
    if (!resources) {
      return;
    }
    if (resources.requests) {
      for (const resourceRequest of resources.requests) {
        const request =
            /** @type {?NetworkRequest} */ (
                self.SDK.networkLog.requests().find(r => r.requestId() === resourceRequest.requestId));
        if (request) {
          // Disconnect the real network request from this issue;
          RelatedIssue.disconnect(request, issue.getCategory(), issue);
        }
      }
    }
  }

  /**
   * @returns {!Iterable}
   */
  aggregatedIssues() {
    return this._aggregatedIssuesByCode.values();
  }

  /**
   * @return {number}
   */
  numberOfAggregatedIssues() {
    return this._aggregatedIssuesByCode.size;
  }
}

/**
 * TODO(chromium:1063765): Change the type (once the protocol/backend changes have landed) to:
 *   !Map>
 *
 * @type {!Map>}
 */
const issueCodeHandlers = new Map([]);

/** @enum {symbol} */
export const Events = {
  AggregatedIssueUpdated: Symbol('AggregatedIssueUpdated'),
  FullUpdateRequired: Symbol('FullUpdateRequired'),
};

SDKModel.register(IssuesModel, Capability.Audits, true);




© 2015 - 2025 Weber Informatics LLC | Privacy Policy