[tor-commits] [Git][tpo/applications/tor-browser][tor-browser-115.1.0esr-13.0-1] 7 commits: fixup! Bug 40933: Add tor-launcher functionality

richard (@richard) git at gitlab.torproject.org
Thu Jul 27 18:01:02 UTC 2023



richard pushed to branch tor-browser-115.1.0esr-13.0-1 at The Tor Project / Applications / Tor Browser


Commits:
66229717 by Pier Angelo Vendrame at 2023-07-27T18:11:55+02:00
fixup! Bug 40933: Add tor-launcher functionality

Bug 41844: Added a couple of wrappers for Onion Auth on
TorProtocolService.

- - - - -
2cde9fc3 by Pier Angelo Vendrame at 2023-07-27T18:11:56+02:00
fixup! Bug 30237: Add v3 onion services client authentication prompt

Bug 41844: Stop using the control port directly

- - - - -
b2cd0ee8 by Pier Angelo Vendrame at 2023-07-27T18:11:57+02:00
fixup! Bug 40933: Add tor-launcher functionality

Small improvements on event registration.

- - - - -
26152fa9 by Pier Angelo Vendrame at 2023-07-27T18:11:57+02:00
fixup! Bug 40933: Add tor-launcher functionality

Bug 41844: Do not use a the control port directly.

Collect the bridge node for the about:preferences#connection page in
TorMonitorService.

Also, move parts of the circuit display to TorMonitorService and
TorProtocolService.

- - - - -
749aeaca by Pier Angelo Vendrame at 2023-07-27T18:11:58+02:00
fixup! Bug 31286: Implementation of bridge, proxy, and firewall settings in about:preferences#connection

Bug 41844: Do not use the control port directly

Do not use the controller in the settings frontend.
Instead, let TorMonitorService collect the first node's fingerprint.

- - - - -
20641450 by Pier Angelo Vendrame at 2023-07-27T18:11:58+02:00
fixup! Bug 3455: Add DomainIsolator, for isolating circuit by domain.

Bug 41844: Do not use the control port directly.

Use TorDomainIsolator also as a backend for the circuit display.

- - - - -
1a8be7b1 by Pier Angelo Vendrame at 2023-07-27T18:11:59+02:00
fixup! Bug 41600: Add a tor circuit display panel.

Bug 41844: Have a separate backend for the tor circuits

Remove the backend stuff from the circuit display.

- - - - -


11 changed files:

- browser/base/content/browser.js
- browser/components/onionservices/content/authPrompt.js
- browser/components/onionservices/content/savedKeysDialog.js
- browser/components/torcircuit/content/torCircuitPanel.js
- browser/components/torpreferences/content/connectionPane.js
- toolkit/components/tor-launcher/TorDomainIsolator.jsm → toolkit/components/tor-launcher/TorDomainIsolator.sys.mjs
- toolkit/components/tor-launcher/TorMonitorService.sys.mjs
- toolkit/components/tor-launcher/TorParsers.sys.mjs
- toolkit/components/tor-launcher/TorProtocolService.sys.mjs
- toolkit/components/tor-launcher/TorStartupService.sys.mjs
- toolkit/components/tor-launcher/moz.build


Changes:

=====================================
browser/base/content/browser.js
=====================================
@@ -66,6 +66,7 @@ ChromeUtils.defineESModuleGetters(this, {
   TabsSetupFlowManager:
     "resource:///modules/firefox-view-tabs-setup-manager.sys.mjs",
   TelemetryEnvironment: "resource://gre/modules/TelemetryEnvironment.sys.mjs",
+  TorDomainIsolator: "resource://gre/modules/TorDomainIsolator.sys.mjs",
   TranslationsParent: "resource://gre/actors/TranslationsParent.sys.mjs",
   UITour: "resource:///modules/UITour.sys.mjs",
   UpdateUtils: "resource://gre/modules/UpdateUtils.sys.mjs",
@@ -100,7 +101,6 @@ XPCOMUtils.defineLazyModuleGetters(this, {
   TorConnect: "resource:///modules/TorConnect.jsm",
   TorConnectState: "resource:///modules/TorConnect.jsm",
   TorConnectTopics: "resource:///modules/TorConnect.jsm",
-  TorDomainIsolator: "resource://gre/modules/TorDomainIsolator.jsm",
   Translation: "resource:///modules/translation/TranslationParent.jsm",
   webrtcUI: "resource:///modules/webrtcUI.jsm",
   ZoomUI: "resource:///modules/ZoomUI.jsm",


=====================================
browser/components/onionservices/content/authPrompt.js
=====================================
@@ -7,6 +7,7 @@
 XPCOMUtils.defineLazyModuleGetters(this, {
   OnionAuthUtil: "chrome://browser/content/onionservices/authUtil.jsm",
   CommonUtils: "resource://services-common/utils.js",
+  TorProtocolService: "resource://gre/modules/TorProtocolService.jsm",
   TorStrings: "resource:///modules/TorStrings.jsm",
 });
 
@@ -192,10 +193,6 @@ const OnionAuthPrompt = (function () {
       let controllerFailureMsg =
         TorStrings.onionServices.authPrompt.failedToSetKey;
       try {
-        let { controller } = ChromeUtils.import(
-          "resource://torbutton/modules/tor-control-port.js"
-        );
-        let torController = await controller();
         // ^(subdomain.)*onionserviceid.onion$ (case-insensitive)
         const onionServiceIdRegExp =
           /^(.*\.)*(?<onionServiceId>[a-z2-7]{56})\.onion$/i;
@@ -206,8 +203,7 @@ const OnionAuthPrompt = (function () {
 
         let checkboxElem = this._getCheckboxElement();
         let isPermanent = checkboxElem && checkboxElem.checked;
-        torController
-          .onionAuthAdd(onionServiceId, base64key, isPermanent)
+        TorProtocolService.onionAuthAdd(onionServiceId, base64key, isPermanent)
           .then(aResponse => {
             // Success! Reload the page.
             this._browser.sendMessageToActor(


=====================================
browser/components/onionservices/content/savedKeysDialog.js
=====================================
@@ -10,8 +10,8 @@ ChromeUtils.defineModuleGetter(
 
 ChromeUtils.defineModuleGetter(
   this,
-  "controller",
-  "resource://torbutton/modules/tor-control-port.js"
+  "TorProtocolService",
+  "resource://gre/modules/TorProtocolService.jsm"
 );
 
 var gOnionServicesSavedKeysDialog = {
@@ -49,11 +49,9 @@ var gOnionServicesSavedKeysDialog = {
       const controllerFailureMsg =
         TorStrings.onionServices.authPreferences.failedToRemoveKey;
       try {
-        const torController = await controller();
-
         // Remove in reverse index order to avoid issues caused by index changes.
         for (let i = indexesToDelete.length - 1; i >= 0; --i) {
-          await this._deleteOneKey(torController, indexesToDelete[i]);
+          await this._deleteOneKey(indexesToDelete[i]);
         }
       } catch (e) {
         if (e.torMessage) {
@@ -127,8 +125,7 @@ var gOnionServicesSavedKeysDialog = {
     try {
       this._tree.view = this;
 
-      const torController = await controller();
-      const keyInfoList = await torController.onionAuthViewKeys();
+      const keyInfoList = await TorProtocolService.onionAuthViewKeys();
       if (keyInfoList) {
         // Filter out temporary keys.
         this._keyInfoList = keyInfoList.filter(aKeyInfo => {
@@ -165,9 +162,9 @@ var gOnionServicesSavedKeysDialog = {
   },
 
   // This method may throw; callers should catch errors.
-  async _deleteOneKey(aTorController, aIndex) {
+  async _deleteOneKey(aIndex) {
     const keyInfoObj = this._keyInfoList[aIndex];
-    await aTorController.onionAuthRemove(keyInfoObj.hsAddress);
+    await TorProtocolService.onionAuthRemove(keyInfoObj.hsAddress);
     this._tree.view.selection.clearRange(aIndex, aIndex);
     this._keyInfoList.splice(aIndex, 1);
     this._tree.rowCountChanged(aIndex + 1, -1);


=====================================
browser/components/torcircuit/content/torCircuitPanel.js
=====================================
@@ -1,18 +1,5 @@
 /* eslint-env mozilla/browser-window */
 
-/**
- * Stores the data associated with a circuit node.
- *
- * @typedef NodeData
- * @property {string[]} ipAddrs - The ip addresses associated with this node.
- * @property {string?} bridgeType - The bridge type for this node, or "" if the
- *   node is a bridge but the type is unknown, or null if this is not a bridge
- *   node.
- * @property {string?} regionCode - An upper case 2-letter ISO3166-1 code for
- *   the first ip address, or null if there is no region. This should also be a
- *   valid BCP47 Region subtag.
- */
-
 /**
  * Data about the current domain and circuit for a xul:browser.
  *
@@ -35,29 +22,6 @@ var gTorCircuitPanel = {
    * @type {Element}
    */
   toolbarButton: null,
-  /**
-   * A list of IDs for "mature" circuits (those that have conveyed a stream).
-   *
-   * @type {string[]}
-   */
-  _knownCircuitIDs: [],
-  /**
-   * Stores the circuit nodes for each SOCKS username/password pair. The keys
-   * are of the form "<username>|<password>".
-   *
-   * @type {Map<string, NodeData[]>}
-   */
-  _credentialsToCircuitNodes: new Map(),
-  /**
-   * Browser data for their currently shown page.
-   *
-   * This data may be stale for a given browser since we only update this data
-   * when loading a new page in the currently selected browser, when switching
-   * tabs, or if we find a new circuit for the current browser.
-   *
-   * @type {WeakMap<MozBrowser, BrowserCircuitData>}
-   */
-  _browserData: new WeakMap(),
   /**
    * The data for the currently shown browser.
    *
@@ -71,6 +35,13 @@ var gTorCircuitPanel = {
    */
   _isActive: false,
 
+  /**
+   * The topic on which circuit changes are broadcast.
+   *
+   * @type {string}
+   */
+  TOR_CIRCUIT_TOPIC: "TorCircuitChange",
+
   /**
    * Initialize the panel.
    */
@@ -86,31 +57,6 @@ var gTorCircuitPanel = {
       maxLogLevelPref: "browser.torcircuitpanel.loglevel",
     });
 
-    const { wait_for_controller } = ChromeUtils.import(
-      "resource://torbutton/modules/tor-control-port.js"
-    );
-    wait_for_controller().then(
-      controller => {
-        if (!this._isActive) {
-          // uninit() was called before resolution.
-          return;
-        }
-        // FIXME: We should be using some dedicated integrated back end to
-        // store circuit information, rather than collecting it all here in the
-        // front end. See tor-browser#41700.
-        controller.watchEvent(
-          "STREAM",
-          streamEvent => streamEvent.StreamStatus === "SENTCONNECT",
-          streamEvent => this._collectCircuit(controller, streamEvent)
-        );
-      },
-      error => {
-        this._log.error(
-          `Not collecting circuits because of an error: ${error.message}`
-        );
-      }
-    );
-
     this.panel = document.getElementById("tor-circuit-panel");
     this._panelElements = {
       heading: document.getElementById("tor-circuit-heading"),
@@ -245,6 +191,9 @@ var gTorCircuitPanel = {
     // Notified of new locations for the currently selected browser (tab) *and*
     // switching selected browser.
     gBrowser.addProgressListener(this._locationListener);
+
+    // Get notifications for circuit changes.
+    Services.obs.addObserver(this, this.TOR_CIRCUIT_TOPIC);
   },
 
   /**
@@ -253,6 +202,17 @@ var gTorCircuitPanel = {
   uninit() {
     this._isActive = false;
     gBrowser.removeProgressListener(this._locationListener);
+    Services.obs.removeObserver(this, this.TOR_CIRCUIT_TOPIC);
+  },
+
+  /**
+   * Observe circuit changes.
+   */
+  observe(subject, topic, data) {
+    if (topic === this.TOR_CIRCUIT_TOPIC) {
+      // TODO: Maybe check if we actually need to do something earlier.
+      this._updateCurrentBrowser();
+    }
   },
 
   /**
@@ -286,109 +246,6 @@ var gTorCircuitPanel = {
     window.openWebLinkIn(this._panelElements.aliasLink.href, where);
   },
 
-  /**
-   * Collect circuit data for the found circuits, to be used later for display.
-   *
-   * @param {controller} controller - The tor controller.
-   * @param {object} streamEvent - The streamEvent for the new circuit.
-   */
-  async _collectCircuit(controller, streamEvent) {
-    const id = streamEvent.CircuitID;
-    if (this._knownCircuitIDs.includes(id)) {
-      return;
-    }
-    this._log.debug(`New streamEvent.CircuitID: ${id}.`);
-    // FIXME: This list grows and is never freed. See tor-browser#41700.
-    this._knownCircuitIDs.push(id);
-    const circuitStatus = (await controller.getInfo("circuit-status"))?.find(
-      circuit => circuit.id === id
-    );
-    if (!circuitStatus?.SOCKS_USERNAME || !circuitStatus?.SOCKS_PASSWORD) {
-      return;
-    }
-    const nodes = await Promise.all(
-      circuitStatus.circuit.map(names =>
-        this._nodeDataForCircuit(controller, names)
-      )
-    );
-    // Remove quotes from the strings.
-    const username = circuitStatus.SOCKS_USERNAME.replace(/^"(.*)"$/, "$1");
-    const password = circuitStatus.SOCKS_PASSWORD.replace(/^"(.*)"$/, "$1");
-    const credentials = `${username}|${password}`;
-    // FIXME: This map grows and is never freed. We cannot simply request this
-    // information when needed because it is no longer available once the
-    // circuit is dropped, even if the web page is still displayed.
-    // See tor-browser#41700.
-    this._credentialsToCircuitNodes.set(credentials, nodes);
-    // Update the circuit in case the current page gains a new circuit whilst
-    // the popup is still open.
-    this._updateCurrentBrowser(credentials);
-  },
-
-  /**
-   * Fetch the node data for the given circuit node.
-   *
-   * @param {controller} controller - The tor controller.
-   * @param {string[]} circuitNodeNames - The names for the circuit node. Only
-   *   the first name, the node id, will be used.
-   *
-   * @returns {NodeData} - The data for this circuit node.
-   */
-  async _nodeDataForCircuit(controller, circuitNodeNames) {
-    // The first "name" in circuitNodeNames is the id.
-    // Remove the leading '$' if present.
-    const id = circuitNodeNames[0].replace(/^\$/, "");
-    let result = { ipAddrs: [], bridgeType: null, regionCode: null };
-    const bridge = (await controller.getConf("bridge"))?.find(
-      foundBridge => foundBridge.ID?.toUpperCase() === id.toUpperCase()
-    );
-    const addrRe = /^\[?([^\]]+)\]?:\d+$/;
-    if (bridge) {
-      result.bridgeType = bridge.type ?? "";
-      // Attempt to get an IP address from bridge address string.
-      const ip = bridge.address.match(addrRe)?.[1];
-      if (ip && !ip.startsWith("0.")) {
-        result.ipAddrs.push(ip);
-      }
-    } else {
-      // Either dealing with a relay, or a bridge whose fingerprint is not saved
-      // in torrc.
-      let statusMap;
-      try {
-        statusMap = await controller.getInfo("ns/id/" + id);
-      } catch {
-        // getInfo will throw if the given id is not a relay.
-        // This probably means we are dealing with a user-provided bridge with
-        // no fingerprint.
-        // We don't know the ip/ipv6 or type, so leave blank.
-        result.bridgeType = "";
-        return result;
-      }
-      if (statusMap.IP && !statusMap.IP.startsWith("0.")) {
-        result.ipAddrs.push(statusMap.IP);
-      }
-      const ip6 = statusMap.IPv6?.match(addrRe)?.[1];
-      if (ip6) {
-        result.ipAddrs.push(ip6);
-      }
-    }
-    if (result.ipAddrs.length) {
-      // Get the country code for the node's IP address.
-      let regionCode;
-      try {
-        // Expect a 2-letter ISO3166-1 code, which should also be a valid BCP47
-        // Region subtag.
-        regionCode = await controller.getInfo(
-          "ip-to-country/" + result.ipAddrs[0]
-        );
-      } catch {}
-      if (regionCode && regionCode !== "??") {
-        result.regionCode = regionCode.toUpperCase();
-      }
-    }
-    return result;
-  },
-
   /**
    * A list of schemes to never show the circuit display for.
    *
@@ -398,71 +255,50 @@ var gTorCircuitPanel = {
    *
    * @type {string[]}
    */
-  // FIXME: Have a back end that handles this instead. See tor-browser#41700.
+  // FIXME: Check if we find a UX to handle some of these cases, and if we
+  // manage to solve some technical issues.
+  // See tor-browser#41700 and tor-browser!699.
   _ignoredSchemes: ["about", "file", "chrome", "resource"],
 
   /**
    * Update the current circuit and domain data for the currently selected
    * browser, possibly changing the UI.
-   *
-   * @param {string?} [matchingCredentials=null] - If given, only update the
-   *   current browser data if the current browser's credentials match.
    */
-  _updateCurrentBrowser(matchingCredentials = null) {
+  _updateCurrentBrowser() {
     const browser = gBrowser.selectedBrowser;
     const domain = TorDomainIsolator.getDomainForBrowser(browser);
+    const nodes = TorDomainIsolator.getCircuit(
+      browser,
+      domain,
+      browser.contentPrincipal.originAttributes.userContextId
+    );
     // We choose the currentURI, which matches what is shown in the URL bar and
     // will match up with the domain.
     // In contrast, documentURI corresponds to the shown page. E.g. it could
     // point to "about:certerror".
     const scheme = browser.currentURI?.scheme;
 
-    let credentials = TorDomainIsolator.getSocksProxyCredentials(
-      domain,
-      browser.contentPrincipal.originAttributes.userContextId
-    );
-    if (credentials) {
-      credentials = `${credentials.username}|${credentials.password}`;
-    }
-
-    if (matchingCredentials && matchingCredentials !== credentials) {
-      // This update was triggered by the circuit update for some other browser
-      // or process.
-      return;
-    }
-
-    let nodes = this._credentialsToCircuitNodes.get(credentials) ?? [];
-
-    const prevData = this._browserData.get(browser);
-    if (
-      prevData &&
-      prevData.domain &&
-      prevData.domain === domain &&
-      prevData.scheme === scheme &&
-      prevData.nodes.length &&
-      !nodes.length
-    ) {
-      // Since this is the same domain, for the same browser, and we used to
-      // have circuit nodes, we *assume* we are re-generating a circuit. So we
-      // keep the old circuit data around for the time being.
-      // FIXME: Have a back end that makes this explicit, rather than an
-      // assumption. See tor-browser#41700.
-      nodes = prevData.nodes;
-      this._log.debug(`Keeping old circuit for ${domain}.`);
-    }
-
-    this._browserData.set(browser, { domain, scheme, nodes });
     if (
       this._currentBrowserData &&
       this._currentBrowserData.domain === domain &&
       this._currentBrowserData.scheme === scheme &&
-      this._currentBrowserData.nodes === nodes
+      this._currentBrowserData.nodes.length === nodes.length &&
+      // If non-null, the fingerprints of the nodes match.
+      (!nodes ||
+        nodes.every(
+          (n, index) =>
+            n.fingerprint === this._currentBrowserData.nodes[index].fingerprint
+        ))
     ) {
       // No change.
+      this._log.debug(
+        "Skipping browser update because the data is already up to date."
+      );
       return;
     }
 
-    this._currentBrowserData = this._browserData.get(browser);
+    this._currentBrowserData = { domain, scheme, nodes };
+    this._log.debug("Updating current browser.", this._currentBrowserData);
 
     if (
       // Schemes where we always want to hide the display.


=====================================
browser/components/torpreferences/content/connectionPane.js
=====================================
@@ -17,6 +17,9 @@ const { TorSettings, TorSettingsTopics, TorSettingsData, TorBridgeSource } =
 const { TorProtocolService } = ChromeUtils.import(
   "resource://gre/modules/TorProtocolService.jsm"
 );
+const { TorMonitorService, TorMonitorTopics } = ChromeUtils.import(
+  "resource://gre/modules/TorMonitorService.jsm"
+);
 
 const { TorConnect, TorConnectTopics, TorConnectState, TorCensorshipLevel } =
   ChromeUtils.import("resource:///modules/TorConnect.jsm");
@@ -144,8 +147,6 @@ const gConnectionPane = (function () {
 
     _internetStatus: InternetStatus.Unknown,
 
-    _controller: null,
-
     _currentBridgeId: null,
 
     // populate xul with strings and cache the relevant elements
@@ -727,9 +728,10 @@ const gConnectionPane = (function () {
       };
       // Use a promise to avoid blocking the population of the page
       // FIXME: Stop using a JSON file, and switch to properties
-      fetch(
+      const annotationPromise = fetch(
         "chrome://browser/content/torpreferences/bridgemoji/annotations.json"
-      ).then(async res => {
+      );
+      annotationPromise.then(async res => {
         const annotations = await res.json();
         const bcp47 = Services.locale.appLocaleAsBCP47;
         const dash = bcp47.indexOf("-");
@@ -749,6 +751,7 @@ const gConnectionPane = (function () {
           ".currently-connected"
         )) {
           card.classList.remove("currently-connected");
+          card.querySelector(selectors.bridges.cardQrGrid).style.height = "";
         }
         if (!this._currentBridgeId) {
           return;
@@ -769,72 +772,17 @@ const gConnectionPane = (function () {
         placeholder.replaceWith(...cards);
         this._checkBridgeCardsHeight();
       };
-      try {
-        const { controller } = ChromeUtils.import(
-          "resource://torbutton/modules/tor-control-port.js"
-        );
-        // Avoid the cache because we set our custom event watcher, and at the
-        // moment, watchers cannot be removed from a controller.
-        controller(true).then(aController => {
-          this._controller = aController;
-          // Getting the circuits may be enough, if we have bootstrapped for a
-          // while, but at the beginning it gives many bridges as connected,
-          // because tor pokes all the bridges to find the best one.
-          // Also, watching circuit events does not work, at the moment, but in
-          // any case, checking the stream has the advantage that we can see if
-          // it really used for a connection, rather than tor having created
-          // this circuit to check if the bridge can be used. We do this by
-          // checking if the stream has SOCKS username, which actually contains
-          // the destination of the stream.
-          // FIXME: We only know the currentBridge *after* a circuit event, but
-          // if the circuit event is sent *before* about:torpreferences is
-          // opened we will miss it. Therefore this approach only works if a
-          // circuit is created after opening about:torconnect. A dedicated
-          // backend outside of about:preferences would help, and could be
-          // shared with gTorCircuitPanel. See tor-browser#41700.
-          this._controller.watchEvent(
-            "STREAM",
-            event =>
-              event.StreamStatus === "SUCCEEDED" && "SOCKS_USERNAME" in event,
-            async event => {
-              const circuitStatuses = await this._controller.getInfo(
-                "circuit-status"
-              );
-              if (!circuitStatuses) {
-                return;
-              }
-              for (const status of circuitStatuses) {
-                if (status.id === event.CircuitID && status.circuit.length) {
-                  // The id in the circuit begins with a $ sign.
-                  const id = status.circuit[0][0].replace(/^\$/, "");
-                  if (id !== this._currentBridgeId) {
-                    const bridge = (
-                      await this._controller.getConf("bridge")
-                    )?.find(
-                      foundBridge =>
-                        foundBridge.ID?.toUpperCase() === id.toUpperCase()
-                    );
-                    if (!bridge) {
-                      // Either there is no bridge, or bridge with no
-                      // fingerprint.
-                      this._currentBridgeId = null;
-                    } else {
-                      this._currentBridgeId = id;
-                    }
-                    this._updateConnectedBridges();
-                  }
-                  break;
-                }
-              }
-            }
-          );
-        });
-      } catch (err) {
-        console.warn(
-          "We could not load torbutton, bridge statuses will not be updated",
-          err
-        );
-      }
+      this._checkConnectedBridge = () => {
+        // TODO: We could make sure TorSettings is in sync by monitoring also
+        // changes of settings. At that point, we could query it, instead of
+        // doing a query over the control port.
+        const bridge = TorMonitorService.currentBridge;
+        if (bridge?.fingerprint !== this._currentBridgeId) {
+          this._currentBridgeId = bridge?.fingerprint ?? null;
+          this._updateConnectedBridges();
+        }
+      };
+      annotationPromise.then(this._checkConnectedBridge.bind(this));
 
       // Add a new bridge
       prefpane.querySelector(selectors.bridges.addHeader).textContent =
@@ -927,6 +875,7 @@ const gConnectionPane = (function () {
       });
 
       Services.obs.addObserver(this, TorConnectTopics.StateChange);
+      Services.obs.addObserver(this, TorMonitorTopics.BridgeChanged);
     },
 
     init() {
@@ -950,11 +899,7 @@ const gConnectionPane = (function () {
       // unregister our observer topics
       Services.obs.removeObserver(this, TorSettingsTopics.SettingChanged);
       Services.obs.removeObserver(this, TorConnectTopics.StateChange);
-
-      if (this._controller !== null) {
-        this._controller.close();
-        this._controller = null;
-      }
+      Services.obs.removeObserver(this, TorMonitorTopics.BridgeChanged);
     },
 
     // whether the page should be present in about:preferences
@@ -985,6 +930,12 @@ const gConnectionPane = (function () {
           this.onStateChange();
           break;
         }
+        case TorMonitorTopics.BridgeChanged: {
+          if (data?.fingerprint !== this._currentBridgeId) {
+            this._checkConnectedBridge();
+          }
+          break;
+        }
       }
     },
 
@@ -1028,7 +979,7 @@ const gConnectionPane = (function () {
     onRemoveAllBridges() {
       TorSettings.bridges.enabled = false;
       TorSettings.bridges.bridge_strings = "";
-      if (TorSettings.bridges.source == TorBridgeSource.BuiltIn) {
+      if (TorSettings.bridges.source === TorBridgeSource.BuiltIn) {
         TorSettings.bridges.builtin_type = "";
       }
       TorSettings.saveToPrefs();


=====================================
toolkit/components/tor-launcher/TorDomainIsolator.jsm → toolkit/components/tor-launcher/TorDomainIsolator.sys.mjs
=====================================
@@ -1,13 +1,14 @@
-// A component for Tor Browser that puts requests from different
-// first party domains on separate Tor circuits.
-
-var EXPORTED_SYMBOLS = ["TorDomainIsolator"];
+/**
+ * A component for Tor Browser that puts requests from different first party
+ * domains on separate Tor circuits.
+ */
 
-const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
-const { XPCOMUtils } = ChromeUtils.import(
-  "resource://gre/modules/XPCOMUtils.jsm"
-);
-const { ConsoleAPI } = ChromeUtils.import("resource://gre/modules/Console.jsm");
+import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
+import { ConsoleAPI } from "resource://gre/modules/Console.sys.mjs";
+import {
+  clearInterval,
+  setInterval,
+} from "resource://gre/modules/Timer.sys.mjs";
 
 const lazy = {};
 
@@ -18,11 +19,10 @@ XPCOMUtils.defineLazyServiceGetters(lazy, {
   ],
 });
 
-ChromeUtils.defineModuleGetter(
-  lazy,
-  "TorProtocolService",
-  "resource://gre/modules/TorProtocolService.jsm"
-);
+ChromeUtils.defineESModuleGetters(lazy, {
+  TorMonitorTopics: "resource://gre/modules/TorMonitorService.sys.mjs",
+  TorProtocolService: "resource://gre/modules/TorProtocolService.sys.mjs",
+});
 
 const logger = new ConsoleAPI({
   prefix: "TorDomainIsolator",
@@ -33,6 +33,12 @@ const logger = new ConsoleAPI({
 // The string to use instead of the domain when it is not known.
 const CATCHALL_DOMAIN = "--unknown--";
 
+// The maximum lifetime for the catch-all circuit in milliseconds.
+// When the catch-all circuit is needed, we check if more than this amount of
+// time has passed since we last changed it nonce, and in case we change it
+// again.
+const CATCHALL_MAX_LIFETIME = 600_000;
+
 // The preference to observe, to know whether isolation should be enabled or
 // disabled.
 const NON_TOR_PROXY_PREF = "extensions.torbutton.use_nontor_proxy";
@@ -40,23 +46,92 @@ const NON_TOR_PROXY_PREF = "extensions.torbutton.use_nontor_proxy";
 // The topic of new identity, to observe to cleanup all the nonces.
 const NEW_IDENTITY_TOPIC = "new-identity-requested";
 
+// The topic on which we broacast circuit change notifications.
+const TOR_CIRCUIT_TOPIC = "TorCircuitChange";
+
+// We have an interval to delete circuits that are not reclaimed by any browser.
+const CLEAR_TIMEOUT = 600_000;
+
+/**
+ * @typedef {string} CircuitId A string that we use to identify a circuit.
+ * Currently, it is a string that combines SOCKS credentials, to make it easier
+ * to use as a map key.
+ * It is not related to Tor's CircuitIDs.
+ */
+/**
+ * @typedef {number} BrowserId
+ */
+/**
+ * @typedef {NodeData[]} CircuitData The data about the nodes, ordered from
+ * guard (or bridge) to exit.
+ */
+/**
+ * @typedef BrowserCircuits Circuits related to a certain combination of
+ * isolators (first-party domain and user context ID, currently).
+ * @property {CircuitId} current The id of the last known circuit that has been
+ * used to fetch data for the isolated context.
+ * @property {CircuitId?} pending The id of the last used circuit for this
+ * isolation context. We might or might not know data about it, yet. But if we
+ * know it, we should move this id into current.
+ */
+
 class TorDomainIsolatorImpl {
-  // A mutable map that records what nonce we are using for each domain.
+  /**
+   * A mutable map that records what nonce we are using for each domain.
+   *
+   * @type {Map<string, string>}
+   */
   #noncesForDomains = new Map();
 
-  // A mutable map that records what nonce we are using for each tab container.
+  /**
+   * A mutable map that records what nonce we are using for each tab container.
+   *
+   * @type {Map<string, string>}
+   */
   #noncesForUserContextId = new Map();
 
-  // A bool that controls if we use SOCKS auth for isolation or not.
+  /**
+   * Tell whether we use SOCKS auth for isolation or not.
+   *
+   * @type {boolean}
+   */
   #isolationEnabled = true;
 
-  // Specifies when the current catch-all circuit was first used
+  /**
+   * Specifies when the current catch-all circuit was first used.
+   *
+   * @type {integer}
+   */
   #catchallDirtySince = Date.now();
 
+  /**
+   * A map that associates circuit ids to the circuit information.
+   *
+   * @type {Map<CircuitId, CircuitData>}
+   */
+  #knownCircuits = new Map();
+
+  /**
+   * A map that associates a certain browser to all the circuits it used or it
+   * is going to use.
+   * The circuits are keyed on the SOCKS username, which we take for granted
+   * being a combination of the first-party domain and the user context id.
+   *
+   * @type {Map<BrowserId, Map<string, BrowserCircuits>>}
+   */
+  #browsers = new Map();
+
+  /**
+   * The handle of the interval we use to cleanup old circuit data.
+   *
+   * @type {number?}
+   */
+  #cleanupIntervalId = null;
+
   /**
    * Initialize the domain isolator.
-   * This function will setup the proxy filter that injects the credentials and
-   * register some observers.
+   * This function will setup the proxy filter that injects the credentials,
+   * register some observers, and setup the cleaning interval.
    */
   init() {
     logger.info("Setup circuit isolation by domain and user context");
@@ -68,14 +143,25 @@ class TorDomainIsolatorImpl {
 
     Services.prefs.addObserver(NON_TOR_PROXY_PREF, this);
     Services.obs.addObserver(this, NEW_IDENTITY_TOPIC);
+    Services.obs.addObserver(this, lazy.TorMonitorTopics.StreamSucceeded);
+
+    this.#cleanupIntervalId = setInterval(
+      this.#clearKnownCircuits.bind(this),
+      CLEAR_TIMEOUT
+    );
   }
 
   /**
-   * Removes the observers added in the initialization.
+   * Removes the observers added in the initialization and stops the cleaning
+   * interval.
    */
   uninit() {
     Services.prefs.removeObserver(NON_TOR_PROXY_PREF, this);
     Services.obs.removeObserver(this, NEW_IDENTITY_TOPIC);
+    Services.obs.removeObserver(this, lazy.TorMonitorTopics.StreamSucceeded);
+    clearInterval(this.#cleanupIntervalId);
+    this.#cleanupIntervalId = null;
+    this.clearIsolation();
   }
 
   enable() {
@@ -89,52 +175,52 @@ class TorDomainIsolatorImpl {
   }
 
   /**
-   * Return the credentials to use as username and password for the SOCKS proxy,
-   * given a certain domain and userContextId. Optionally, create them.
+   * Get the last circuit used in a certain browser.
+   * The returned data is created when the circuit is first seen, therefore it
+   * could be stale (i.e., the circuit might not be available anymore).
    *
-   * @param {string} firstPartyDomain The first party domain associated to the requests
-   * @param {string} userContextId The context ID associated to the request
-   * @param {bool} create Whether to create the nonce, if it is not available
-   * @returns {object|null} Either the credential, or null if we do not have them and create is
-   * false.
+   * @param {MozBrowser} browser The browser to get data for
+   * @param {string} domain The first party domain we want to get the circuit
+   * for
+   * @param {number} userContextId The user context domain we want to get the
+   * circuit for
+   * @returns {NodeData[]} The node data, or an empty array if we do not have
+   * data for the requested key.
    */
-  getSocksProxyCredentials(firstPartyDomain, userContextId, create = false) {
-    if (!this.#noncesForDomains.has(firstPartyDomain)) {
-      if (!create) {
-        return null;
-      }
-      const nonce = this.#nonce();
-      logger.info(`New nonce for first party ${firstPartyDomain}: ${nonce}`);
-      this.#noncesForDomains.set(firstPartyDomain, nonce);
+  getCircuit(browser, domain, userContextId) {
+    const username = this.#makeUsername(domain, userContextId);
+    const circuits = this.#browsers.get(browser.browserId)?.get(username);
+    // This is the only place where circuit data can go out, so the only place
+    // where it makes a difference to check whether the pending circuit is still
+    // pending, or it has actually got data.
+    const pending = this.#knownCircuits.get(circuits?.pending);
+    if (pending?.length) {
+      circuits.current = circuits.pending;
+      circuits.pending = null;
+      return pending;
     }
-    if (!this.#noncesForUserContextId.has(userContextId)) {
-      if (!create) {
-        return null;
-      }
-      const nonce = this.#nonce();
-      logger.info(`New nonce for userContextId ${userContextId}: ${nonce}`);
-      this.#noncesForUserContextId.set(userContextId, nonce);
-    }
-    return {
-      username: this.#makeUsername(firstPartyDomain, userContextId),
-      password:
-        this.#noncesForDomains.get(firstPartyDomain) +
-        this.#noncesForUserContextId.get(userContextId),
-    };
+    // TODO: At this point we already know if we expect a circuit change for
+    // this key: (circuit?.pending && !pending). However, we do not consume this
+    // data yet in the frontend, so do not send it for now.
+    return this.#knownCircuits.get(circuits?.current) ?? [];
   }
 
   /**
    * Create a new nonce for the FP domain of the selected browser and reload the
    * tab with a new circuit.
    *
-   * @param {object} browser Should be the gBrowser from the context of the
-   * caller
+   * @param {object} globalBrowser Should be the gBrowser from the context of
+   * the caller
    */
-  newCircuitForBrowser(browser) {
-    const firstPartyDomain = getDomainForBrowser(browser.selectedBrowser);
+  newCircuitForBrowser(globalBrowser) {
+    const browser = globalBrowser.selectedBrowser;
+    const firstPartyDomain = getDomainForBrowser(browser);
     this.#newCircuitForDomain(firstPartyDomain);
-    // TODO: How to properly handle the user context? Should we use
-    // (domain, userContextId) pairs, instead of concatenating nonces?
+    const { username, password } = this.#getSocksProxyCredentials(
+      firstPartyDomain,
+      browser.contentPrincipal.originAttributes.userContextId
+    );
+    this.#trackBrowser(browser, username, password);
     browser.reloadWithFlags(Ci.nsIWebNavigation.LOAD_FLAGS_BYPASS_CACHE);
   }
 
@@ -147,12 +233,15 @@ class TorDomainIsolatorImpl {
 
     // Per-domain and per contextId nonces are stored in maps, so simply clear
     // them.
+    // Notice that the catch-all circuit is included in #noncesForDomains, so we
+    // are implicilty cleaning it. Should this change, we should change its
+    // nonce explicitly here.
     this.#noncesForDomains.clear();
     this.#noncesForUserContextId.clear();
+    this.#catchallDirtySince = Date.now();
 
-    // Force a rotation on the next catch-all circuit use by setting the
-    // creation time to the epoch.
-    this.#catchallDirtySince = 0;
+    this.#knownCircuits.clear();
+    this.#browsers.clear();
   }
 
   async observe(subject, topic, data) {
@@ -173,55 +262,20 @@ class TorDomainIsolatorImpl {
         logger.error("Could not send the newnym command", e);
         // TODO: What UX to use here? See tor-browser#41708
       }
+    } else if (topic === lazy.TorMonitorTopics.StreamSucceeded) {
+      const { username, password, circuit } = subject.wrappedJSObject;
+      this.#updateCircuit(username, password, circuit);
     }
   }
 
   /**
-   * Setup a filter that for every HTTPChannel, replaces the default SOCKS proxy
-   * with one that authenticates to the SOCKS server (the tor client process)
-   * with a username (the first party domain and userContextId) and a nonce
-   * password.
-   * Tor provides a separate circuit for each username+password combination.
+   * Setup a filter that for every HTTPChannel.
    */
   #setupProxyFilter() {
-    const filterFunction = (aChannel, aProxy) => {
-      if (!this.#isolationEnabled) {
-        return aProxy;
-      }
-      try {
-        const channel = aChannel.QueryInterface(Ci.nsIChannel);
-        let firstPartyDomain =
-          channel.loadInfo.originAttributes.firstPartyDomain;
-        const userContextId = channel.loadInfo.originAttributes.userContextId;
-        if (firstPartyDomain === "") {
-          firstPartyDomain = CATCHALL_DOMAIN;
-          if (Date.now() - this.#catchallDirtySince > 1000 * 10 * 60) {
-            logger.info(
-              "tor catchall circuit has been dirty for over 10 minutes. Rotating."
-            );
-            this.#newCircuitForDomain(CATCHALL_DOMAIN);
-            this.#catchallDirtySince = Date.now();
-          }
-        }
-        const replacementProxy = this.#applySocksProxyCredentials(
-          aProxy,
-          firstPartyDomain,
-          userContextId
-        );
-        logger.debug(
-          `Requested ${channel.URI.spec} via ${replacementProxy.username}:${replacementProxy.password}`
-        );
-        return replacementProxy;
-      } catch (e) {
-        logger.error("Error while setting a new proxy", e);
-        return null;
-      }
-    };
-
     lazy.ProtocolProxyService.registerChannelFilter(
       {
-        applyFilter(aChannel, aProxy, aCallback) {
-          aCallback.onProxyFilterResult(filterFunction(aChannel, aProxy));
+        applyFilter: (aChannel, aProxy, aCallback) => {
+          aCallback.onProxyFilterResult(this.#proxyFilter(aChannel, aProxy));
         },
       },
       0
@@ -229,33 +283,96 @@ class TorDomainIsolatorImpl {
   }
 
   /**
-   * Takes a proxyInfo object (originalProxy) and returns a new proxyInfo
-   * object with the same properties, except the username is set to the
-   * the domain and userContextId, and the password is a nonce.
+   * Replaces the default SOCKS proxy with one that authenticates to the SOCKS
+   * server (the tor client process) with a username (the first party domain and
+   * userContextId) and a nonce password.
+   * Tor provides a separate circuit for each username+password combination.
+   *
+   * @param {nsIChannel} aChannel The channel we are setting the proxy for
+   * @param {nsIProxyInfo} aProxy The original proxy
+   * @returns {nsIProxyInfo} The new proxy to use
    */
-  #applySocksProxyCredentials(originalProxy, domain, userContextId) {
-    const proxy = originalProxy.QueryInterface(Ci.nsIProxyInfo);
-    const { username, password } = this.getSocksProxyCredentials(
-      domain,
-      userContextId,
-      true
-    );
-    return lazy.ProtocolProxyService.newProxyInfoWithAuth(
-      "socks",
-      proxy.host,
-      proxy.port,
-      username,
-      password,
-      "", // aProxyAuthorizationHeader
-      "", // aConnectionIsolationKey
-      proxy.flags,
-      proxy.failoverTimeout,
-      proxy.failoverProxy
-    );
+  #proxyFilter(aChannel, aProxy) {
+    if (!this.#isolationEnabled) {
+      return aProxy;
+    }
+    try {
+      const channel = aChannel.QueryInterface(Ci.nsIChannel);
+      let firstPartyDomain = channel.loadInfo.originAttributes.firstPartyDomain;
+      const userContextId = channel.loadInfo.originAttributes.userContextId;
+      if (!firstPartyDomain) {
+        firstPartyDomain = CATCHALL_DOMAIN;
+        if (Date.now() - this.#catchallDirtySince > CATCHALL_MAX_LIFETIME) {
+          logger.info(
+            "tor catchall circuit has reached its maximum lifetime. Rotating."
+          );
+          this.#newCircuitForDomain(CATCHALL_DOMAIN);
+        }
+      }
+      const { username, password } = this.#getSocksProxyCredentials(
+        firstPartyDomain,
+        userContextId
+      );
+      const browser = this.#getBrowserForChannel(channel);
+      if (browser) {
+        this.#trackBrowser(browser, username, password);
+      }
+      logger.debug(`Requested ${channel.URI.spec} via ${username}:${password}`);
+      const proxy = aProxy.QueryInterface(Ci.nsIProxyInfo);
+      return lazy.ProtocolProxyService.newProxyInfoWithAuth(
+        "socks",
+        proxy.host,
+        proxy.port,
+        username,
+        password,
+        "", // aProxyAuthorizationHeader
+        "", // aConnectionIsolationKey
+        proxy.flags,
+        proxy.failoverTimeout,
+        proxy.failoverProxy
+      );
+    } catch (e) {
+      logger.error("Error while setting a new proxy", e);
+      return null;
+    }
+  }
+
+  /**
+   * Return the credentials to use as username and password for the SOCKS proxy,
+   * given a certain domain and userContextId.
+   * A new random password will be created if not available yet.
+   *
+   * @param {string} firstPartyDomain The first party domain associated to the
+   * requests
+   * @param {number} userContextId The context ID associated to the request
+   * @returns {object} The credentials
+   */
+  #getSocksProxyCredentials(firstPartyDomain, userContextId) {
+    if (!this.#noncesForDomains.has(firstPartyDomain)) {
+      const nonce = this.#nonce();
+      logger.info(`New nonce for first party ${firstPartyDomain}: ${nonce}`);
+      this.#noncesForDomains.set(firstPartyDomain, nonce);
+    }
+    if (!this.#noncesForUserContextId.has(userContextId)) {
+      const nonce = this.#nonce();
+      logger.info(`New nonce for userContextId ${userContextId}: ${nonce}`);
+      this.#noncesForUserContextId.set(userContextId, nonce);
+    }
+    // TODO: How to properly handle the user-context? Should we use
+    // (domain, userContextId) pairs, instead of concatenating nonces?
+    return {
+      username: this.#makeUsername(firstPartyDomain, userContextId),
+      password:
+        this.#noncesForDomains.get(firstPartyDomain) +
+        this.#noncesForUserContextId.get(userContextId),
+    };
   }
 
   /**
    * Combine the needed data into a username for the proxy.
+   *
+   * @param {string} domain The first-party domain associated to the request
+   * @param {integer} userContextId The userContextId associated to the request
    */
   #makeUsername(domain, userContextId) {
     if (!domain) {
@@ -264,12 +381,26 @@ class TorDomainIsolatorImpl {
     return `${domain}:${userContextId}`;
   }
 
+  /**
+   * Combine SOCKS username and password into a string to use as ID.
+   *
+   * @param {string} username The SOCKS username
+   * @param {string} password The SOCKS password
+   * @returns {CircuitId} A string that combines username and password and can
+   * be used for map lookups.
+   */
+  #credentialsToId(username, password) {
+    return `${username}|${password}`;
+  }
+
   /**
    * Generate a new 128 bit random tag.
    *
    * Strictly speaking both using a cryptographic entropy source and using 128
    * bits of entropy for the tag are likely overkill, as correct behavior only
    * depends on how unlikely it is for there to be a collision.
+   *
+   * @returns {string} The random nonce
    */
   #nonce() {
     return Array.from(crypto.getRandomValues(new Uint8Array(16)), byte =>
@@ -279,12 +410,18 @@ class TorDomainIsolatorImpl {
 
   /**
    * Re-generate the nonce for a certain domain.
+   *
+   * @param {string?} domain The first-party domain to re-create the nonce for.
+   * If empty or null, the catchall domain will be used.
    */
   #newCircuitForDomain(domain) {
     if (!domain) {
       domain = CATCHALL_DOMAIN;
     }
     this.#noncesForDomains.set(domain, this.#nonce());
+    if (domain === CATCHALL_DOMAIN) {
+      this.#catchallDirtySince = Date.now();
+    }
     logger.info(
       `New domain isolation for ${domain}: ${this.#noncesForDomains.get(
         domain
@@ -296,6 +433,8 @@ class TorDomainIsolatorImpl {
    * Re-generate the nonce for a userContextId.
    *
    * Currently, this function is not hooked to anything.
+   *
+   * @param {integer} userContextId The userContextId to re-create the nonce for
    */
   #newCircuitForUserContextId(userContextId) {
     this.#noncesForUserContextId.set(userContextId, this.#nonce());
@@ -305,13 +444,182 @@ class TorDomainIsolatorImpl {
       )}`
     );
   }
+
+  /**
+   * Try to extract a browser from a channel.
+   *
+   * @param {nsIChannel} channel The channel to extract the browser from
+   * @returns {MozBrowser?} The browser the channel is associated to
+   */
+  #getBrowserForChannel(channel) {
+    const browsers =
+      channel.loadInfo.browsingContext?.topChromeWindow?.gBrowser.browsers;
+    if (!browsers || !channel.loadInfo.browsingContext?.browserId) {
+      return null;
+    }
+    for (const browser of browsers) {
+      if (browser.browserId === channel.loadInfo.browsingContext.browserId) {
+        logger.debug(
+          "Matched browser with browserId",
+          channel.loadInfo.browsingContext.browserId
+        );
+        return browser;
+      }
+    }
+    // Expected to arrive here for example for the update checker.
+    // If we find a way to check that, we could raise the level to a warn.
+    logger.debug("Browser not matched", channel);
+    return null;
+  }
+
+  /**
+   * Associate the SOCKS credentials to a browser.
+   * If needed (the browser is associated for the first time, or it was already
+   * known but its credential changed), notify the related circuit display.
+   *
+   * @param {MozBrowser} browser The browser to track
+   * @param {string} username The SOCKS username
+   * @param {string} password The SOCKS password
+   */
+  #trackBrowser(browser, username, password) {
+    let browserCircuits = this.#browsers.get(browser.browserId);
+    if (!browserCircuits) {
+      browserCircuits = new Map();
+      this.#browsers.set(browser.browserId, browserCircuits);
+    }
+    const circuitIds = browserCircuits.get(username) ?? {};
+    const id = this.#credentialsToId(username, password);
+    if (circuitIds.current === id) {
+      // The circuit with these credentials was already built (we already knew
+      // its nodes, or we would not have promoted it to the current circuit).
+      // We do not need to do anything else, because we cannot detect a change
+      // of nodes here.
+      return;
+    }
+
+    logger.debug(
+      `Found new credentials ${username} ${password} for browser`,
+      browser
+    );
+    const circuit = this.#knownCircuits.get(id);
+    if (circuit?.length) {
+      circuitIds.current = id;
+      if (circuitIds.pending === id) {
+        circuitIds.pending = null;
+      }
+      browserCircuits.set(username, circuitIds);
+      // FIXME: We only notify the circuit display when we have a change that
+      // involves circuits whose nodes are known, for now. We need to resolve a
+      // few other techical problems (e.g., associate the circuit to the
+      // document?) and develop a UX with some animation to notify the circuit
+      // display more often.
+      // See tor-browser#41700 and tor-browser!699.
+      // In any case, notify the circuit display only after the internal map has
+      // been updated.
+      this.#notifyCircuitDisplay();
+    } else if (circuitIds.pending !== id) {
+      // We do not have node data, so we store that we might need to track this.
+      // Otherwise, when a circuit is ready, we do not know which browser was it
+      // used for.
+      circuitIds.pending = id;
+      browserCircuits.set(username, circuitIds);
+    }
+  }
+
+  /**
+   * Update a circuit, and notify the related circuit displays if it changed.
+   *
+   * This function is called when a certain stream has succeeded and so we can
+   * associate its SOCKS credential to the circuit it is using.
+   * We receive only the fingerprints of the circuit nodes, but they are enough
+   * to check if the circuit has changed. If it has, we also get the nodes'
+   * information through the control port.
+   *
+   * @param {string} username The SOCKS username
+   * @param {string} password The SOCKS password
+   * @param {NodeFingerprint[]} circuit The fingerprints of the nodes that
+   * compose the circuit
+   */
+  async #updateCircuit(username, password, circuit) {
+    const id = this.#credentialsToId(username, password);
+    let data = this.#knownCircuits.get(id) ?? [];
+    // Should we modify the lower layer to send a circuit identifier, instead?
+    if (
+      circuit.length === data.length &&
+      circuit.every((id, index) => id === data[index].fingerprint)
+    ) {
+      return;
+    }
+
+    data = await Promise.all(
+      circuit.map(fingerprint =>
+        lazy.TorProtocolService.getNodeInfo(fingerprint)
+      )
+    );
+    this.#knownCircuits.set(id, data);
+    // We know that something changed, but we cannot know if anyone is
+    // interested in this change. So, we have to notify all the possible
+    // consumers of the data in any case.
+    // Not being specific and let them check if they need to do something allows
+    // us to keep a simpler structure.
+    this.#notifyCircuitDisplay();
+  }
+
+  /**
+   * Broadcast a notification when a circuit changed, or a browser is changing
+   * circuit (which might happen also in case of navigation).
+   */
+  #notifyCircuitDisplay() {
+    Services.obs.notifyObservers(null, TOR_CIRCUIT_TOPIC);
+  }
+
+  /**
+   * Clear the known circuit information, when they are not needed anymore.
+   *
+   * We keep circuit data around for a while. We decouple it from the underlying
+   * tor circuit management in case the user clicks on the circuit display when
+   * circuit has long gone.
+   * However, data accumulate during a session. So, since we store all the
+   * browsers that used a circuit anyway, every now and then we check if we
+   * still know browsers using a certain circuits. If there are not, we forget
+   * about it.
+   *
+   * This function is run by an interval.
+   */
+  #clearKnownCircuits() {
+    logger.info("Running the circuit cleanup");
+    const windows = [];
+    const enumerator = Services.wm.getEnumerator("navigator:browser");
+    while (enumerator.hasMoreElements()) {
+      windows.push(enumerator.getNext());
+    }
+    const browsers = windows
+      .flatMap(win => win.gBrowser.browsers.map(b => b.browserId))
+      .filter(id => this.#browsers.has(id));
+    this.#browsers = new Map(browsers.map(id => [id, this.#browsers.get(id)]));
+    this.#knownCircuits = new Map(
+      Array.from(this.#browsers.values(), circuits =>
+        Array.from(circuits.values(), ids => {
+          const r = [];
+          const current = this.#knownCircuits.get(ids.current);
+          if (current) {
+            r.push([ids.current, current]);
+          }
+          const pending = this.#knownCircuits.get(ids.pending);
+          if (pending) {
+            r.push([ids.pending, pending]);
+          }
+          return r;
+        })
+      ).flat(2)
+    );
+  }
 }
 
 /**
  * Get the first party domain for a certain browser.
  *
- * @param browser The browser to get the FP-domain for.
- *
+ * @param {MozBrowser} browser The browser to get the FP-domain for.
  * Please notice that it should be gBrowser.selectedBrowser, because
  * browser.documentURI is the actual shown page, and might be an error page.
  * In this case, we rely on currentURI, which for gBrowser is an alias of
@@ -358,6 +666,6 @@ function getDomainForBrowser(browser) {
   return fpd;
 }
 
-const TorDomainIsolator = new TorDomainIsolatorImpl();
+export const TorDomainIsolator = new TorDomainIsolatorImpl();
 // Reduce global vars pollution
 TorDomainIsolator.getDomainForBrowser = getDomainForBrowser;


=====================================
toolkit/components/tor-launcher/TorMonitorService.sys.mjs
=====================================
@@ -19,6 +19,10 @@ ChromeUtils.defineModuleGetter(
   "resource://torbutton/modules/tor-control-port.js"
 );
 
+ChromeUtils.defineESModuleGetters(lazy, {
+  TorProtocolService: "resource://gre/modules/TorProtocolService.sys.mjs",
+});
+
 const logger = new ConsoleAPI({
   maxLogLevel: "warn",
   maxLogLevelPref: "browser.tor_monitor_service.log_level",
@@ -37,12 +41,34 @@ const TorTopics = Object.freeze({
   ProcessRestarted: "TorProcessRestarted",
 });
 
+export const TorMonitorTopics = Object.freeze({
+  BridgeChanged: "TorBridgeChanged",
+  StreamSucceeded: "TorStreamSucceeded",
+});
+
 const ControlConnTimings = Object.freeze({
   initialDelayMS: 25, // Wait 25ms after the process has started, before trying to connect
   maxRetryMS: 10000, // Retry at most every 10 seconds
   timeoutMS: 5 * 60 * 1000, // Wait at most 5 minutes for tor to start
 });
 
+/**
+ * From control-spec.txt:
+ *   CircuitID = 1*16 IDChar
+ *   IDChar = ALPHA / DIGIT
+ *   Currently, Tor only uses digits, but this may change.
+ *
+ * @typedef {string} CircuitID
+ */
+/**
+ * The fingerprint of a node.
+ * From control-spec.txt:
+ *   Fingerprint = "$" 40*HEXDIG
+ * However, we do not keep the $ in our structures.
+ *
+ * @typedef {string} NodeFingerprint
+ */
+
 /**
  * This service monitors an existing Tor instance, or starts one, if needed, and
  * then starts monitoring it.
@@ -52,7 +78,7 @@ const ControlConnTimings = Object.freeze({
  */
 export const TorMonitorService = {
   _connection: null,
-  _eventsToMonitor: Object.freeze(["STATUS_CLIENT", "NOTICE", "WARN", "ERR"]),
+  _eventHandlers: {},
   _torLog: [], // Array of objects with date, type, and msg properties.
   _startTimeout: null,
 
@@ -64,6 +90,28 @@ export const TorMonitorService = {
 
   _inited: false,
 
+  /**
+   * Stores the nodes of a circuit. Keys are cicuit IDs, and values are the node
+   * fingerprints.
+   *
+   * Theoretically, we could hook this map up to the new identity notification,
+   * but in practice it does not work. Tor pre-builds circuits, and the NEWNYM
+   * signal does not affect them. So, we might end up using a circuit that was
+   * built before the new identity but not yet used. If we cleaned the map, we
+   * risked of not having the data about it.
+   *
+   * @type {Map<CircuitID, NodeFingerprint[]>}
+   */
+  _circuits: new Map(),
+  /**
+   * The last used bridge, or null if bridges are not in use or if it was not
+   * possible to detect the bridge. This needs the user to have specified bridge
+   * lines with fingerprints to work.
+   *
+   * @type {NodeFingerprint?}
+   */
+  _currentBridge: null,
+
   // Public methods
 
   // Starts Tor, if needed, and starts monitoring for events
@@ -72,14 +120,28 @@ export const TorMonitorService = {
       return;
     }
     this._inited = true;
+
+    // We always liten to these events, because they are needed for the circuit
+    // display.
+    this._eventHandlers = new Map([
+      ["CIRC", this._processCircEvent.bind(this)],
+      ["STREAM", this._processStreamEvent.bind(this)],
+    ]);
+
     if (this.ownsTorDaemon) {
+      // When we own the tor daemon, we listen to more events, that are used
+      // for about:torconnect or for showing the logs in the settings page.
+      this._eventHandlers.set("STATUS_CLIENT", (_eventType, lines) =>
+        this._processBootstrapStatus(lines[0], false)
+      );
+      this._eventHandlers.set("NOTICE", this._processLog.bind(this));
+      this._eventHandlers.set("WARN", this._processLog.bind(this));
+      this._eventHandlers.set("ERR", this._processLog.bind(this));
       this._controlTor();
     } else {
-      logger.info(
-        "Not starting the event monitor, as we do not own the Tor daemon."
-      );
+      this._startEventMonitor();
     }
-    logger.debug("TorMonitorService initialized");
+    logger.info("TorMonitorService initialized");
   },
 
   // Closes the connection that monitors for events.
@@ -153,6 +215,18 @@ export const TorMonitorService = {
     return !!this._connection;
   },
 
+  /**
+   * Return the data about the current bridge, if any, or null.
+   * We can detect bridge only when the configured bridge lines include the
+   * fingerprints.
+   *
+   * @returns {NodeData?} The node information, or null if the first node
+   * is not a bridge, or no circuit has been opened, yet.
+   */
+  get currentBridge() {
+    return this._currentBridge;
+  },
+
   // Private methods
 
   async _startProcess() {
@@ -272,7 +346,7 @@ export const TorMonitorService = {
 
     // TODO: optionally monitor INFO and DEBUG log messages.
     let reply = await conn.sendCommand(
-      "SETEVENTS " + this._eventsToMonitor.join(" ")
+      "SETEVENTS " + Array.from(this._eventHandlers.keys()).join(" ")
     );
     reply = TorParsers.parseCommandResponse(reply);
     if (!TorParsers.commandSucceeded(reply)) {
@@ -281,14 +355,10 @@ export const TorMonitorService = {
       return false;
     }
 
-    // FIXME: At the moment it is not possible to start the event monitor
-    // when we do start the tor process. So, does it make sense to keep this
-    // control?
     if (this._torProcess) {
       this._torProcess.connectionWorked();
     }
-
-    if (!TorLauncherUtil.shouldOnlyConfigureTor) {
+    if (this.ownsTorDaemon && !TorLauncherUtil.shouldOnlyConfigureTor) {
       try {
         await this._takeTorOwnership(conn);
       } catch (e) {
@@ -297,7 +367,31 @@ export const TorMonitorService = {
     }
 
     this._connection = conn;
-    this._waitForEventData();
+
+    for (const [type, callback] of this._eventHandlers.entries()) {
+      this._monitorEvent(type, callback);
+    }
+
+    // Populate the circuit map already, in case we are connecting to an
+    // external tor daemon.
+    try {
+      const reply = await this._connection.sendCommand(
+        "GETINFO circuit-status"
+      );
+      const lines = reply.split(/\r?\n/);
+      if (lines.shift() === "250+circuit-status=") {
+        for (const line of lines) {
+          if (line === ".") {
+            break;
+          }
+          // _processCircEvent processes only one line at a time
+          this._processCircEvent("CIRC", [line]);
+        }
+      }
+    } catch (e) {
+      logger.warn("Could not populate the initial circuit map", e);
+    }
+
     return true;
   },
 
@@ -318,65 +412,49 @@ export const TorMonitorService = {
     }
   },
 
-  _waitForEventData() {
-    if (!this._connection) {
-      return;
-    }
-    logger.debug("Start watching events:", this._eventsToMonitor);
+  _monitorEvent(type, callback) {
+    logger.info(`Watching events of type ${type}.`);
     let replyObj = {};
-    for (const torEvent of this._eventsToMonitor) {
-      this._connection.watchEvent(
-        torEvent,
-        null,
-        line => {
-          if (!line) {
-            return;
-          }
-          logger.debug("Event response: ", line);
-          const isComplete = TorParsers.parseReplyLine(line, replyObj);
-          if (isComplete) {
-            this._processEventReply(replyObj);
-            replyObj = {};
-          }
-        },
-        true
-      );
-    }
+    this._connection.watchEvent(
+      type,
+      null,
+      line => {
+        if (!line) {
+          return;
+        }
+        logger.debug("Event response: ", line);
+        const isComplete = TorParsers.parseReplyLine(line, replyObj);
+        if (!isComplete || replyObj._parseError || !replyObj.lineArray.length) {
+          return;
+        }
+        const reply = replyObj;
+        replyObj = {};
+        if (reply.statusCode !== TorStatuses.EventNotification) {
+          logger.error("Unexpected event status code:", reply.statusCode);
+          return;
+        }
+        if (!reply.lineArray[0].startsWith(`${type} `)) {
+          logger.error("Wrong format for the first line:", reply.lineArray[0]);
+          return;
+        }
+        reply.lineArray[0] = reply.lineArray[0].substring(type.length + 1);
+        try {
+          callback(type, reply.lineArray);
+        } catch (e) {
+          logger.error("Exception while handling an event", reply, e);
+        }
+      },
+      true
+    );
   },
 
-  _processEventReply(aReply) {
-    if (aReply._parseError || !aReply.lineArray.length) {
-      return;
-    }
-
-    if (aReply.statusCode !== TorStatuses.EventNotification) {
-      logger.warn("Unexpected event status code:", aReply.statusCode);
-      return;
-    }
-
-    // TODO: do we need to handle multiple lines?
-    const s = aReply.lineArray[0];
-    const idx = s.indexOf(" ");
-    if (idx === -1) {
-      return;
-    }
-    const eventType = s.substring(0, idx);
-    const msg = s.substring(idx + 1).trim();
-
-    if (eventType === "STATUS_CLIENT") {
-      this._processBootstrapStatus(msg, false);
-      return;
-    } else if (!this._eventsToMonitor.includes(eventType)) {
-      logger.debug(`Dropping unlistened event ${eventType}`);
-      return;
-    }
-
-    if (eventType === "WARN" || eventType === "ERR") {
+  _processLog(type, lines) {
+    if (type === "WARN" || type === "ERR") {
       // Notify so that Copy Log can be enabled.
       Services.obs.notifyObservers(null, TorTopics.HasWarnOrErr);
     }
 
-    const now = new Date();
+    const date = new Date();
     const maxEntries = Services.prefs.getIntPref(
       "extensions.torlauncher.max_tor_log_entries",
       1000
@@ -384,8 +462,10 @@ export const TorMonitorService = {
     if (maxEntries > 0 && this._torLog.length >= maxEntries) {
       this._torLog.splice(0, 1);
     }
-    this._torLog.push({ date: now, type: eventType, msg });
-    const logString = `Tor ${eventType}: ${msg}`;
+
+    const msg = lines.join("\n");
+    this._torLog.push({ date, type, msg });
+    const logString = `Tor ${type}: ${msg}`;
     logger.info(logString);
   },
 
@@ -461,8 +541,108 @@ export const TorMonitorService = {
     }
   },
 
+  async _processCircEvent(_type, lines) {
+    const builtEvent =
+      /^(?<CircuitID>[a-zA-Z0-9]{1,16})\sBUILT\s(?<Path>(?:,?\$[0-9a-fA-F]{40}(?:~[a-zA-Z0-9]{1,19})?)+)/.exec(
+        lines[0]
+      );
+    const closedEvent = /^(?<ID>[a-zA-Z0-9]{1,16})\sCLOSED/.exec(lines[0]);
+    if (builtEvent) {
+      const fp = /\$([0-9a-fA-F]{40})/g;
+      const nodes = Array.from(builtEvent.groups.Path.matchAll(fp), g =>
+        g[1].toUpperCase()
+      );
+      this._circuits.set(builtEvent.groups.CircuitID, nodes);
+      // Ignore circuits of length 1, that are used, for example, to probe
+      // bridges. So, only store them, since we might see streams that use them,
+      // but then early-return.
+      if (nodes.length === 1) {
+        return;
+      }
+      // In some cases, we might already receive SOCKS credentials in the line.
+      // However, this might be a problem with onion services: we get also a
+      // 4-hop circuit that we likely do not want to show to the user,
+      // especially because it is used only temporarily, and it would need a
+      // technical explaination.
+      // this._checkCredentials(lines[0], nodes);
+      if (this._currentBridge?.fingerprint !== nodes[0]) {
+        const nodeInfo = await lazy.TorProtocolService.getNodeInfo(nodes[0]);
+        let notify = false;
+        if (nodeInfo?.bridgeType) {
+          logger.info(`Bridge changed to ${nodes[0]}`);
+          this._currentBridge = nodeInfo;
+          notify = true;
+        } else if (this._currentBridge) {
+          logger.info("Bridges disabled");
+          this._currentBridge = null;
+          notify = true;
+        }
+        if (notify) {
+          Services.obs.notifyObservers(
+            null,
+            TorMonitorTopics.BridgeChanged,
+            this._currentBridge
+          );
+        }
+      }
+    } else if (closedEvent) {
+      this._circuits.delete(closedEvent.groups.ID);
+    }
+  },
+
+  _processStreamEvent(_type, lines) {
+    // The first block is the stream ID, which we do not need at the moment.
+    const succeeedEvent =
+      /^[a-zA-Z0-9]{1,16}\sSUCCEEDED\s(?<CircuitID>[a-zA-Z0-9]{1,16})/.exec(
+        lines[0]
+      );
+    if (!succeeedEvent) {
+      return;
+    }
+    const circuit = this._circuits.get(succeeedEvent.groups.CircuitID);
+    if (!circuit) {
+      logger.error(
+        "Seen a STREAM SUCCEEDED with an unknown circuit. Not notifying observers.",
+        lines[0]
+      );
+      return;
+    }
+    this._checkCredentials(lines[0], circuit);
+  },
+
+  /**
+   * Check if a STREAM or CIRC response line contains SOCKS_USERNAME and
+   * SOCKS_PASSWORD. In case, notify observers that we could associate a certain
+   * circuit to these credentials.
+   *
+   * @param {string} line The circ or stream line to check
+   * @param {NodeFingerprint[]} circuit The fingerprints of the nodes in the
+   * circuit.
+   */
+  _checkCredentials(line, circuit) {
+    const username = /SOCKS_USERNAME=("(?:[^"\\]|\\.)*")/.exec(line);
+    const password = /SOCKS_PASSWORD=("(?:[^"\\]|\\.)*")/.exec(line);
+    if (!username || !password) {
+      return;
+    }
+    Services.obs.notifyObservers(
+      {
+        wrappedJSObject: {
+          username: TorParsers.unescapeString(username[1]),
+          password: TorParsers.unescapeString(password[1]),
+          circuit,
+        },
+      },
+      TorMonitorTopics.StreamSucceeded
+    );
+  },
+
   _shutDownEventMonitor() {
-    this._connection?.close();
+    try {
+      this._connection?.close();
+    } catch (e) {
+      logger.error("Could not close the connection to the control port", e);
+    }
     this._connection = null;
     if (this._startTimeout !== null) {
       clearTimeout(this._startTimeout);


=====================================
toolkit/components/tor-launcher/TorParsers.sys.mjs
=====================================
@@ -181,12 +181,12 @@ export const TorParsers = Object.freeze({
       return aStr;
     }
     const escaped = aStr
-      .replace("\\", "\\\\")
-      .replace('"', '\\"')
-      .replace("\n", "\\n")
-      .replace("\r", "\\r")
-      .replace("\t", "\\t")
-      .replace(/[^\x20-\x7e]+/g, text => {
+      .replaceAll("\\", "\\\\")
+      .replaceAll('"', '\\"')
+      .replaceAll("\n", "\\n")
+      .replaceAll("\r", "\\r")
+      .replaceAll("\t", "\\t")
+      .replaceAll(/[^\x20-\x7e]+/g, text => {
         const encoder = new TextEncoder();
         return Array.from(
           encoder.encode(text),


=====================================
toolkit/components/tor-launcher/TorProtocolService.sys.mjs
=====================================
@@ -40,6 +40,20 @@ const logger = new ConsoleAPI({
   prefix: "TorProtocolService",
 });
 
+/**
+ * Stores the data associated with a circuit node.
+ *
+ * @typedef NodeData
+ * @property {string} fingerprint The node fingerprint.
+ * @property {string[]} ipAddrs - The ip addresses associated with this node.
+ * @property {string?} bridgeType - The bridge type for this node, or "" if the
+ *   node is a bridge but the type is unknown, or null if this is not a bridge
+ *   node.
+ * @property {string?} regionCode - An upper case 2-letter ISO3166-1 code for
+ *   the first ip address, or null if there is no region. This should also be a
+ *   valid BCP47 Region subtag.
+ */
+
 // Manage the connection to tor's control port, to update its settings and query
 // other useful information.
 //
@@ -188,6 +202,89 @@ export const TorProtocolService = {
     return TorParsers.parseReply(cmd, keyword, response);
   },
 
+  async getBridges() {
+    // Ideally, we would not need this function, because we should be the one
+    // setting them with TorSettings. However, TorSettings is not notified of
+    // change of settings. So, asking tor directly with the control connection
+    // is the most reliable way of getting the configured bridges, at the
+    // moment. Also, we are using this for the circuit display, which should
+    // work also when we are not configuring the tor daemon, but just using it.
+    return this._withConnection(conn => {
+      return conn.getConf("bridge");
+    });
+  },
+
+  /**
+   * Returns tha data about a relay or a bridge.
+   *
+   * @param {string} id The fingerprint of the node to get data about
+   * @returns {NodeData}
+   */
+  async getNodeInfo(id) {
+    return this._withConnection(async conn => {
+      const node = {
+        fingerprint: id,
+        ipAddrs: [],
+        bridgeType: null,
+        regionCode: null,
+      };
+      const bridge = (await conn.getConf("bridge"))?.find(
+        foundBridge => foundBridge.ID?.toUpperCase() === id.toUpperCase()
+      );
+      const addrRe = /^\[?([^\]]+)\]?:\d+$/;
+      if (bridge) {
+        node.bridgeType = bridge.type ?? "";
+        // Attempt to get an IP address from bridge address string.
+        const ip = bridge.address.match(addrRe)?.[1];
+        if (ip && !ip.startsWith("0.")) {
+          node.ipAddrs.push(ip);
+        }
+      } else {
+        // Either dealing with a relay, or a bridge whose fingerprint is not
+        // saved in torrc.
+        const info = await conn.getInfo(`ns/id/${id}`);
+        if (info.IP && !info.IP.startsWith("0.")) {
+          node.ipAddrs.push(info.IP);
+        }
+        const ip6 = info.IPv6?.match(addrRe)?.[1];
+        if (ip6) {
+          node.ipAddrs.push(ip6);
+        }
+      }
+      if (node.ipAddrs.length) {
+        // Get the country code for the node's IP address.
+        let regionCode;
+        try {
+          // Expect a 2-letter ISO3166-1 code, which should also be a valid
+          // BCP47 Region subtag.
+          regionCode = await conn.getInfo("ip-to-country/" + node.ipAddrs[0]);
+        } catch {}
+        if (regionCode && regionCode !== "??") {
+          node.regionCode = regionCode.toUpperCase();
+        }
+      }
+      return node;
+    });
+  },
+
+  async onionAuthAdd(hsAddress, b64PrivateKey, isPermanent) {
+    return this._withConnection(conn => {
+      return conn.onionAuthAdd(hsAddress, b64PrivateKey, isPermanent);
+    });
+  },
+
+  async onionAuthRemove(hsAddress) {
+    return this._withConnection(conn => {
+      return conn.onionAuthRemove(hsAddress);
+    });
+  },
+
+  async onionAuthViewKeys() {
+    return this._withConnection(conn => {
+      return conn.onionAuthViewKeys();
+    });
+  },
+
   // TODO: transform the following 4 functions in getters. At the moment they
   // are also used in torbutton.
 
@@ -630,6 +727,16 @@ export const TorProtocolService = {
     }
   },
 
+  async _withConnection(func) {
+    // TODO: Make more robust?
+    const conn = await this._getConnection();
+    try {
+      return await func(conn);
+    } finally {
+      this._returnConnection();
+    }
+  },
+
   // If aConn is omitted, the cached connection is closed.
   _closeConnection() {
     if (this._controlConnection) {


=====================================
toolkit/components/tor-launcher/TorStartupService.sys.mjs
=====================================
@@ -3,6 +3,7 @@ const lazy = {};
 // We will use the modules only when the profile is loaded, so prefer lazy
 // loading
 ChromeUtils.defineESModuleGetters(lazy, {
+  TorDomainIsolator: "resource://gre/modules/TorDomainIsolator.sys.mjs",
   TorLauncherUtil: "resource://gre/modules/TorLauncherUtil.sys.mjs",
   TorMonitorService: "resource://gre/modules/TorMonitorService.sys.mjs",
   TorProtocolService: "resource://gre/modules/TorProtocolService.sys.mjs",
@@ -19,12 +20,6 @@ ChromeUtils.defineModuleGetter(
   "resource:///modules/TorSettings.jsm"
 );
 
-ChromeUtils.defineModuleGetter(
-  lazy,
-  "TorDomainIsolator",
-  "resource://gre/modules/TorDomainIsolator.jsm"
-);
-
 /* Browser observer topis */
 const BrowserTopics = Object.freeze({
   ProfileAfterChange: "profile-after-change",


=====================================
toolkit/components/tor-launcher/moz.build
=====================================
@@ -1,6 +1,6 @@
 EXTRA_JS_MODULES += [
     "TorBootstrapRequest.sys.mjs",
-    "TorDomainIsolator.jsm",
+    "TorDomainIsolator.sys.mjs",
     "TorLauncherUtil.sys.mjs",
     "TorMonitorService.sys.mjs",
     "TorParsers.sys.mjs",



View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/bdda467fbaec13a0dc9ad40997d0261f7396f5da...1a8be7b1e2fd495282857ea7b17972ed8892d133

-- 
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/bdda467fbaec13a0dc9ad40997d0261f7396f5da...1a8be7b1e2fd495282857ea7b17972ed8892d133
You're receiving this email because of your account on gitlab.torproject.org.


-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.torproject.org/pipermail/tor-commits/attachments/20230727/99b20161/attachment-0001.htm>


More information about the tor-commits mailing list