forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			215 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			215 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* This Source Code Form is subject to the terms of the Mozilla Public
 | 
						|
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 | 
						|
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | 
						|
 | 
						|
import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
 | 
						|
 | 
						|
const lazy = {};
 | 
						|
 | 
						|
ChromeUtils.defineESModuleGetters(lazy, {
 | 
						|
  PingCentre: "resource:///modules/PingCentre.sys.mjs",
 | 
						|
  Region: "resource://gre/modules/Region.sys.mjs",
 | 
						|
});
 | 
						|
 | 
						|
// Endpoint base URL for Structured Ingestion
 | 
						|
XPCOMUtils.defineLazyPreferenceGetter(
 | 
						|
  lazy,
 | 
						|
  "structuredIngestionEndpointBase",
 | 
						|
  "browser.newtabpage.activity-stream.telemetry.structuredIngestion.endpoint",
 | 
						|
  ""
 | 
						|
);
 | 
						|
const NAMESPACE_CONTEXUAL_SERVICES = "contextual-services";
 | 
						|
 | 
						|
// PingCentre client to send custom pings
 | 
						|
ChromeUtils.defineLazyGetter(lazy, "pingcentre", () => {
 | 
						|
  return new lazy.PingCentre({ topic: "contextual-services" });
 | 
						|
});
 | 
						|
 | 
						|
// `contextId` is a unique identifier used by Contextual Services
 | 
						|
const CONTEXT_ID_PREF = "browser.contextual-services.contextId";
 | 
						|
ChromeUtils.defineLazyGetter(lazy, "contextId", () => {
 | 
						|
  let _contextId = Services.prefs.getStringPref(CONTEXT_ID_PREF, null);
 | 
						|
  if (!_contextId) {
 | 
						|
    _contextId = String(Services.uuid.generateUUID());
 | 
						|
    Services.prefs.setStringPref(CONTEXT_ID_PREF, _contextId);
 | 
						|
  }
 | 
						|
  return _contextId;
 | 
						|
});
 | 
						|
 | 
						|
export const CONTEXTUAL_SERVICES_PING_TYPES = {
 | 
						|
  TOPSITES_IMPRESSION: "topsites-impression",
 | 
						|
  TOPSITES_SELECTION: "topsites-click",
 | 
						|
  QS_BLOCK: "quicksuggest-block",
 | 
						|
  QS_IMPRESSION: "quicksuggest-impression",
 | 
						|
  QS_SELECTION: "quicksuggest-click",
 | 
						|
};
 | 
						|
 | 
						|
export var PartnerLinkAttribution = {
 | 
						|
  /**
 | 
						|
   * Sends an attribution request to an anonymizing proxy.
 | 
						|
   *
 | 
						|
   * @param {string} targetURL
 | 
						|
   *   The URL we are routing through the anonmyzing proxy.
 | 
						|
   * @param {string} source
 | 
						|
   *   The source of the anonmized request, e.g. "urlbar".
 | 
						|
   * @param {string} [campaignID]
 | 
						|
   *   The campaign ID for attribution. This should be a valid path on the
 | 
						|
   *   anonymizing proxy. For example, if `campaignID` was `foo`, we'd send an
 | 
						|
   *   attribution request to https://topsites.mozilla.com/cid/foo.
 | 
						|
   *   Optional. If it's not provided, we default to the topsites campaign.
 | 
						|
   */
 | 
						|
  async makeRequest({ targetURL, source, campaignID }) {
 | 
						|
    let partner = targetURL.match(/^https?:\/\/(?:www.)?([^.]*)/)[1];
 | 
						|
 | 
						|
    function record(method, objectString) {
 | 
						|
      recordTelemetryEvent({
 | 
						|
        method,
 | 
						|
        objectString,
 | 
						|
        value: partner,
 | 
						|
      });
 | 
						|
    }
 | 
						|
    record("click", source);
 | 
						|
 | 
						|
    let attributionUrl = Services.prefs.getStringPref(
 | 
						|
      "browser.partnerlink.attributionURL"
 | 
						|
    );
 | 
						|
    if (!attributionUrl) {
 | 
						|
      record("attribution", "abort");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // The default campaign is topsites.
 | 
						|
    if (!campaignID) {
 | 
						|
      campaignID = Services.prefs.getStringPref(
 | 
						|
        "browser.partnerlink.campaign.topsites"
 | 
						|
      );
 | 
						|
    }
 | 
						|
    attributionUrl = attributionUrl + campaignID;
 | 
						|
    let result = await sendRequest(attributionUrl, source, targetURL);
 | 
						|
    record("attribution", result ? "success" : "failure");
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Makes a request to the attribution URL for a search engine search.
 | 
						|
   *
 | 
						|
   * @param {nsISearchEngine} engine
 | 
						|
   *   The search engine to save the attribution for.
 | 
						|
   * @param {nsIURI} targetUrl
 | 
						|
   *   The target URL to filter and include in the attribution.
 | 
						|
   */
 | 
						|
  async makeSearchEngineRequest(engine, targetUrl) {
 | 
						|
    let cid;
 | 
						|
    if (engine.attribution?.cid) {
 | 
						|
      cid = engine.attribution.cid;
 | 
						|
    } else if (engine.sendAttributionRequest) {
 | 
						|
      cid = Services.prefs.getStringPref(
 | 
						|
        "browser.partnerlink.campaign.topsites"
 | 
						|
      );
 | 
						|
    } else {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let searchUrlQueryParamName = engine.searchUrlQueryParamName;
 | 
						|
    if (!searchUrlQueryParamName) {
 | 
						|
      console.error("makeSearchEngineRequest can't find search terms key");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let url = targetUrl;
 | 
						|
    if (typeof url == "string") {
 | 
						|
      url = Services.io.newURI(url);
 | 
						|
    }
 | 
						|
 | 
						|
    let targetParams = new URLSearchParams(url.query);
 | 
						|
    if (!targetParams.has(searchUrlQueryParamName)) {
 | 
						|
      console.error("makeSearchEngineRequest can't remove target search terms");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let attributionUrl = Services.prefs.getStringPref(
 | 
						|
      "browser.partnerlink.attributionURL",
 | 
						|
      ""
 | 
						|
    );
 | 
						|
    attributionUrl = attributionUrl + cid;
 | 
						|
 | 
						|
    targetParams.delete(searchUrlQueryParamName);
 | 
						|
    let strippedTargetUrl = `${url.prePath}${url.filePath}`;
 | 
						|
    let newParams = targetParams.toString();
 | 
						|
    if (newParams) {
 | 
						|
      strippedTargetUrl += "?" + newParams;
 | 
						|
    }
 | 
						|
 | 
						|
    await sendRequest(attributionUrl, "searchurl", strippedTargetUrl);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Sends a Contextual Services ping to the Mozilla data pipeline.
 | 
						|
   *
 | 
						|
   * Note:
 | 
						|
   *   * All Contextual Services pings are sent as custom pings
 | 
						|
   *     (https://docs.telemetry.mozilla.org/cookbooks/new_ping.html#sending-a-custom-ping)
 | 
						|
   *
 | 
						|
   *   * The full event list can be found at https://github.com/mozilla-services/mozilla-pipeline-schemas
 | 
						|
   *     under the "contextual-services" namespace
 | 
						|
   *
 | 
						|
   * @param {object} payload
 | 
						|
   *   The ping payload to be sent to the Mozilla Structured Ingestion endpoint
 | 
						|
   * @param {String} pingType
 | 
						|
   *   The ping type. Must be one of CONTEXTUAL_SERVICES_PING_TYPES
 | 
						|
   */
 | 
						|
  sendContextualServicesPing(payload, pingType) {
 | 
						|
    if (!Object.values(CONTEXTUAL_SERVICES_PING_TYPES).includes(pingType)) {
 | 
						|
      console.error("Invalid Contextual Services ping type");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    const endpoint = makeEndpointUrl(pingType, "1");
 | 
						|
    payload.context_id = lazy.contextId;
 | 
						|
    lazy.pingcentre.sendStructuredIngestionPing(
 | 
						|
      payload,
 | 
						|
      endpoint,
 | 
						|
      NAMESPACE_CONTEXUAL_SERVICES
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Gets the underlying PingCentre client, only used for tests.
 | 
						|
   */
 | 
						|
  get _pingCentre() {
 | 
						|
    return lazy.pingcentre;
 | 
						|
  },
 | 
						|
};
 | 
						|
 | 
						|
async function sendRequest(attributionUrl, source, targetURL) {
 | 
						|
  const request = new Request(attributionUrl);
 | 
						|
  request.headers.set("X-Region", lazy.Region.home);
 | 
						|
  request.headers.set("X-Source", source);
 | 
						|
  request.headers.set("X-Target-URL", targetURL);
 | 
						|
  const response = await fetch(request);
 | 
						|
  return response.ok;
 | 
						|
}
 | 
						|
 | 
						|
function recordTelemetryEvent({ method, objectString, value }) {
 | 
						|
  Services.telemetry.setEventRecordingEnabled("partner_link", true);
 | 
						|
  Services.telemetry.recordEvent("partner_link", method, objectString, value);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Makes a new endpoint URL for a ping submission. Note that each submission
 | 
						|
 * to Structured Ingesttion requires a new endpoint. See more details about
 | 
						|
 * the specs:
 | 
						|
 *
 | 
						|
 * https://docs.telemetry.mozilla.org/concepts/pipeline/http_edge_spec.html?highlight=docId#postput-request
 | 
						|
 *
 | 
						|
 * @param {String} pingType
 | 
						|
 *   The ping type. Must be one of CONTEXTUAL_SERVICES_PING_TYPES
 | 
						|
 * @param {String} version
 | 
						|
 *   The schema version of the ping.
 | 
						|
 */
 | 
						|
function makeEndpointUrl(pingType, version) {
 | 
						|
  // Structured Ingestion does not support the UUID generated by gUUIDGenerator.
 | 
						|
  // Stripping off the leading and trailing braces to make it happy.
 | 
						|
  const docID = Services.uuid.generateUUID().toString().slice(1, -1);
 | 
						|
  const extension = `${NAMESPACE_CONTEXUAL_SERVICES}/${pingType}/${version}/${docID}`;
 | 
						|
  return `${lazy.structuredIngestionEndpointBase}/${extension}`;
 | 
						|
}
 |