tor-commits
Threads by month
- ----- 2026 -----
- May
- April
- March
- February
- January
- ----- 2025 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2024 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2023 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2022 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2021 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2020 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2019 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2018 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2017 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2016 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2015 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2014 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2013 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2012 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- January
- ----- 2011 -----
- December
- November
- October
- September
- August
- July
- June
- May
- April
- March
- February
- 1 participants
- 215222 discussions
[Git][tpo/applications/tor-browser][base-browser-146.0a1-16.0-2] 7 commits: fixup! BB 41803: Add some developer tools for working on tor-browser.
by henry (ï¼ henry) 11 Dec '25
by henry (ï¼ henry) 11 Dec '25
11 Dec '25
henry pushed to branch base-browser-146.0a1-16.0-2 at The Tor Project / Applications / Tor Browser
Commits:
5138b860 by Henry Wilkes at 2025-12-11T14:43:50+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Make git_get return the stdout string, rather than a list.
Add git_lines to generate lines.
- - - - -
3aff7d98 by Henry Wilkes at 2025-12-11T14:43:51+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Use raw diff to get list of file changes.
- - - - -
5ba0688d by Henry Wilkes at 2025-12-11T14:43:52+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Add type annotations and parameter documentation.
- - - - -
08c1d176 by Henry Wilkes at 2025-12-11T14:43:53+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Make the argcomplete module optional.
- - - - -
ebac78a8 by Henry Wilkes at 2025-12-11T14:43:54+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Use function caching instead of global variables.
- - - - -
12776f40 by Henry Wilkes at 2025-12-11T14:43:55+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Fetch FIREFOX_ tags from the remote if they are missing.
- - - - -
18b3272d by Henry Wilkes at 2025-12-11T14:43:56+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Improve the auto-fixup/auto-commit command.
The auto-fixup command was renamed to auto-commit. It now also handles:
1. Already staged changes.
2. Untracked/added files.
3. Removed files.
4. Renamed files.
5. Allowing the user to create a new commit.
- - - - -
1 changed file:
- tools/base_browser/tb-dev
Changes:
=====================================
tools/base_browser/tb-dev
=====================================
@@ -6,6 +6,7 @@ Useful tools for working on tor-browser repository.
import argparse
import atexit
+import functools
import json
import os
import re
@@ -14,8 +15,15 @@ import sys
import tempfile
import termios
import urllib.request
+from collections.abc import Callable, Iterable, Iterator
+from types import ModuleType
+from typing import Any, NotRequired, TypedDict, TypeVar
-import argcomplete
+argcomplete: None | ModuleType = None
+try:
+ import argcomplete
+except ImportError:
+ pass
GIT_PATH = "/usr/bin/git"
UPSTREAM_URLS = {
@@ -36,9 +44,14 @@ class TbDevException(Exception):
pass
-def git_run(args, check=True, env=None):
+def git_run(
+ args: list[str], check: bool = True, env: None | dict[str, str] = None
+) -> None:
"""
Run a git command with output sent to stdout.
+ :param args: The arguments to pass to git.
+ :param check: Whether to check for success.
+ :param env: Optional environment to set.
"""
if env is not None:
tmp_env = dict(os.environ)
@@ -51,46 +64,122 @@ def git_run(args, check=True, env=None):
raise TbDevException(str(err)) from err
-def git_get(args):
+def git_run_pager(
+ args: list[str] | None = None,
+ arg_sequence: Iterable[list[str]] | None = None,
+ pager_prefix: None | str = None,
+) -> None:
"""
- Run a git command with each non-empty line returned in a list.
+ Run a sequence of git commands with the output concatenated and sent to the
+ git pager.
+ :param args: The arguments to pass to git, or `None` if a sequence is desired.
+ :param arg_sequence: A sequence representing several git commands.
+ :param pager_prefix: An optional text to send to the pager first.
+ """
+ if arg_sequence is None:
+ if args is not None:
+ arg_sequence = (args,)
+ else:
+ raise ValueError("Missing `arg_sequence` or `args`")
+ elif args is not None:
+ raise ValueError("Unexpected both args and arg_sequence")
+
+ pager = git_get(["var", "GIT_PAGER"])
+ if not pager:
+ raise TbDevException("Missing a GIT_PAGER")
+ command = [pager]
+ if os.path.basename(pager) == "less":
+ # Show colours.
+ command.append("-R")
+
+ pager_process = subprocess.Popen(command, stdin=subprocess.PIPE, text=True)
+ assert pager_process.stdin is not None
+
+ if pager_prefix is not None:
+ pager_process.stdin.write(pager_prefix)
+ pager_process.stdin.flush()
+
+ for git_args in arg_sequence:
+ subprocess.run(
+ [GIT_PATH, "--no-pager", *git_args], check=False, stdout=pager_process.stdin
+ )
+
+ pager_process.stdin.close()
+
+ status = pager_process.wait()
+ if status != 0:
+ raise TbDevException(f"git pager {pager} exited with status {status}")
+
+
+def git_get(args: list[str], strip: bool = True, check: bool = True) -> str:
+ """
+ Return the output from a git command.
+ :param args: The arguments to send to git.
+ :param strip: Whether to strip the whitespace from the output.
+ :param check: Whether to check for success.
+ :returns: The stdout.
"""
try:
git_process = subprocess.run(
- [GIT_PATH, *args], text=True, stdout=subprocess.PIPE, check=True
+ [GIT_PATH, *args], text=True, stdout=subprocess.PIPE, check=check
)
except subprocess.CalledProcessError as err:
raise TbDevException(str(err)) from err
- return [line for line in git_process.stdout.split("\n") if line]
+ ret = git_process.stdout
+ if strip:
+ ret = ret.strip()
+ return ret
-local_root = None
+def git_lines(args: list[str]) -> Iterator[str]:
+ """
+ Yields the non-empty lines returned by the git command.
+ :param args: The arguments to send to git.
+ :yield: The lines.
+ """
+ for line in git_get(args, strip=False).split("\n"):
+ if not line:
+ continue
+ yield line
+
+
+def git_path_args(path_iter: Iterable[str]) -> Iterator[str]:
+ """
+ Generate the trailing arguments to specify paths in git commands, includes
+ the "--" separator just before the paths.
+ :param path_iter: The paths that should be passed in.
+ :yields: The git arguments.
+ """
+ yield "--"
+ for path in path_iter:
+ yield f":(literal){path}"
-def get_local_root():
+(a)functools.cache
+def get_local_root() -> str:
"""
Get the path for the tor-browser root directory.
+ :returns: The local root.
"""
- global local_root
- if local_root is None:
- try:
- # Make sure we have a matching remote in this git repository.
- if get_upstream_details()["is-browser-repo"]:
- local_root = git_get(["rev-parse", "--show-toplevel"])[0]
- else:
- local_root = ""
- except TbDevException:
- local_root = ""
- return local_root
+ try:
+ # Make sure we have a matching remote in this git repository.
+ if get_upstream_details()["is-browser-repo"] == "True":
+ return git_get(["rev-parse", "--show-toplevel"])
+ else:
+ return ""
+ except TbDevException:
+ return ""
-def determine_upstream_details():
+(a)functools.cache
+def get_upstream_details() -> dict[str, str]:
"""
- Determine details about the upstream.
+ Get details about the upstream repository.
+ :returns: The details.
"""
remote_urls = {
- remote: git_get(["remote", "get-url", remote])[0]
- for remote in git_get(["remote"])
+ remote: git_get(["remote", "get-url", remote])
+ for remote in git_lines(["remote"])
}
matches = {
@@ -102,7 +191,7 @@ def determine_upstream_details():
}
is_browser_repo = len(matches) > 0
- details = {"is-browser-repo": is_browser_repo}
+ details = {"is-browser-repo": str(is_browser_repo)}
origin_remote_repo = matches.get("origin", None)
upstream_remote_repo = matches.get("upstream", None)
@@ -125,31 +214,30 @@ def determine_upstream_details():
return details
-cached_upstream_details = None
-
-
-def get_upstream_details():
- """
- Get details about the upstream repository.
- """
- global cached_upstream_details
- if cached_upstream_details is None:
- cached_upstream_details = determine_upstream_details()
- return cached_upstream_details
-
-
class Reference:
"""Represents a git reference to a commit."""
- def __init__(self, name, commit):
- self.name = name
+ _REFS_REGEX = re.compile(r"refs/[a-z]+/")
+
+ def __init__(self, full_name: str, commit: str) -> None:
+ """
+ :param full_name: The full reference name. E.g. "refs/tags/MyTag".
+ :param commit: The commit hash for the commit this reference points to.
+ """
+ match = self.__class__._REFS_REGEX.match(full_name)
+ if not match:
+ raise ValueError(f"Invalid reference name {full_name}")
+ self.full_name = full_name
+ self.name = full_name[match.end() :]
self.commit = commit
-def get_refs(ref_type, name_start):
+def get_refs(ref_type: str, name_start: str) -> Iterator[Reference]:
"""
- Get a list of references that match the given 'ref_type' ("tag" or "remote"
- or "head") that starts with the given 'name_start'.
+ Get a list of references that match the given conditions.
+ :param ref_type: The ref type to search for ("tag" or "remote" or "head").
+ :param name_start: The ref name start to match against.
+ :yield: The matching references.
"""
if ref_type == "tag":
ref_start = "refs/tags/"
@@ -163,56 +251,83 @@ def get_refs(ref_type, name_start):
fstring = "%(*objectname),%(objectname),%(refname)"
pattern = f"{ref_start}{name_start}**"
- def line_to_ref(line):
+ def line_to_ref(line: str) -> Reference:
[objectname_reference, objectname, ref_name] = line.split(",", 2)
# For annotated tags, the objectname_reference is non-empty and points
# to an actual commit.
# For remotes, heads and lightweight tags, the objectname_reference will
# be empty and objectname will point directly to the commit.
- return Reference(
- ref_name.replace(ref_start, "", 1), objectname_reference or objectname
- )
+ return Reference(ref_name, objectname_reference or objectname)
- return [
+ return (
line_to_ref(line)
- for line in git_get(["for-each-ref", f"--format={fstring}", pattern])
- ]
+ for line in git_lines(["for-each-ref", f"--format={fstring}", pattern])
+ )
-def get_nearest_ref(ref_type, name_start, search_from):
+def get_firefox_ref(search_from: str) -> Reference:
"""
- Search backwards from the 'search_from' commit to find the first commit
- that matches the given 'ref_type' that starts with the given 'name_start'.
+ Search for the commit that comes from firefox.
+ :param search_from: The commit to search backwards from.
+ :returns: The firefox reference.
"""
- ref_list = get_refs(ref_type, name_start)
+ # Only search a limited history that should include the FIREFOX_ tag.
+ search_commits = [c for c in git_lines(["rev-list", "-1000", search_from])]
+
+ firefox_tag_prefix = "FIREFOX_"
- for commit in git_get(["rev-list", "-1000", search_from]):
- for ref in ref_list:
+ existing_tags = list(get_refs("tag", firefox_tag_prefix))
+ for commit in search_commits:
+ for ref in existing_tags:
if commit == ref.commit:
return ref
- raise TbDevException(f"No {name_start} commit found in the last 1000 commits")
-
-
-def get_firefox_ref(search_from):
+ # Might just need to fetch tags from the remote.
+ upstream = get_upstream_details().get("remote", None)
+ if upstream:
+ remote_ref: None | Reference = None
+ search_index = len(search_commits)
+ # Search the remote for a tag that is in our history.
+ # We want to avoid triggering a long fetch, so we just want to grab the
+ # tag that already points to a commit in our history.
+ for line in git_lines(
+ ["ls-remote", upstream, f"refs/tags/{firefox_tag_prefix}*"]
+ ):
+ objectname, name = line.split("\t", 1)
+ for index in range(search_index):
+ if search_commits[index] == objectname:
+ # Remove trailing "^{}" for commits pointed to by
+ # annotated tags.
+ remote_ref = Reference(re.sub(r"\^\{\}$", "", name), objectname)
+ # Only continue to search for references that are even
+ # closer to `search_from`.
+ search_index = index
+ break
+ if remote_ref is not None:
+ # Get a local copy of just this tag.
+ git_run(["fetch", "--no-tags", upstream, "tag", remote_ref.name])
+ return ref
+
+ raise TbDevException("Unable to find FIREFOX_ tag")
+
+
+def get_upstream_tracking_branch(search_from: str) -> str:
"""
- Search backwards from the 'search_from' commit to find the commit that comes
- from firefox.
+ :param search_from: The commit reference.
+ :returns: The upstream branch reference name.
"""
- return get_nearest_ref("tag", "FIREFOX_", search_from)
-
-
-def get_upstream_tracking_branch(search_from):
- return git_get(["rev-parse", "--abbrev-ref", f"{search_from}@{{upstream}}"])[0]
+ return git_get(["rev-parse", "--abbrev-ref", f"{search_from}@{{upstream}}"])
-def get_upstream_basis_commit(search_from):
+def get_upstream_basis_commit(search_from: str) -> str:
"""
Get the first common ancestor of search_from that is also in its upstream
branch.
+ :param search_from: The commit reference.
+ :returns: The upstream commit hash.
"""
upstream_branch = get_upstream_tracking_branch(search_from)
- commit = git_get(["merge-base", search_from, upstream_branch])[0]
+ commit = git_get(["merge-base", search_from, upstream_branch])
# Verify that the upstream commit shares the same firefox basis. Otherwise,
# this would indicate that the upstream is on an early or later FIREFOX
# base.
@@ -226,26 +341,82 @@ def get_upstream_basis_commit(search_from):
return commit
-def get_changed_files(from_commit, staged=False):
+class FileChange:
+ """Represents a git change to a commit."""
+
+ def __init__(self, status: str, path: str, new_path: str) -> None:
+ """
+ :param status: The file change status used within git diff. E.g. "M" for
+ modified, or "D" for deleted.
+ :param path: The source file path.
+ :param new_path: The file path after the change.
+ """
+ self.status = status
+ self.path = path
+ self.new_path = new_path
+
+
+RAW_DIFF_PATH_PATTERN = r"(?P<path>[^\0]*)\0"
+RAW_DIFF_LINE_REGEX = re.compile(
+ r":[0-7]+ [0-7]+ [0-9a-f]+ [0-9a-f]+ (?P<status>[ADMTUXRC])[0-9]*\0"
+ + RAW_DIFF_PATH_PATTERN
+)
+RAW_DIFF_PATH_REGEX = re.compile(RAW_DIFF_PATH_PATTERN)
+
+
+def parse_raw_diff_line(raw_output: str) -> tuple[FileChange, int]:
"""
- Get a list of filenames relative to the current working directory that have
+ Parse the --raw diff output from git.
+ :param raw_output: The raw output.
+ :returns: The change for this line, and the offset for the end of the raw
+ diff line.
+ """
+ match = RAW_DIFF_LINE_REGEX.match(raw_output)
+ if not match:
+ raise ValueError(f"Invalid raw output: {raw_output[:50]}...")
+ path = os.path.relpath(os.path.join(get_local_root(), match.group("path")))
+ status = match.group("status")
+ if status in ("R", "C"):
+ match = RAW_DIFF_PATH_REGEX.match(raw_output, pos=match.end())
+ if not match:
+ raise ValueError(f"Invalid raw output for rename: {raw_output[:50]}...")
+ new_path = os.path.relpath(os.path.join(get_local_root(), match.group("path")))
+ else:
+ new_path = path
+
+ return FileChange(status, path, new_path), match.end()
+
+
+def get_changed_files(
+ from_commit: None | str = None, staged: bool = False
+) -> Iterator[FileChange]:
+ """
+ Get a list of file changes relative to the current working directory that have
been changed since 'from_commit' (non-inclusive).
+ :param from_commit: The commit to compare against, otherwise use the git
+ diff default.
+ :param staged: Whether to limit the diff to staged changes.
+ :yield: The file changes.
"""
- args = ["diff"]
+ args = ["diff", "-z", "--raw"]
if staged:
args.append("--staged")
- args.append("--name-only")
- args.append(from_commit)
- return [
- os.path.relpath(os.path.join(get_local_root(), filename))
- for filename in git_get(args)
- ]
+ if from_commit:
+ args.append(from_commit)
+ raw_output = git_get(args, strip=False)
+ while raw_output:
+ file_change, end = parse_raw_diff_line(raw_output)
+ yield file_change
+ raw_output = raw_output[end:]
-def file_contains(filename, regex):
+def file_contains(filename: str, regex: re.Pattern[str]) -> bool:
"""
Return whether the file is a utf-8 text file containing the regular
expression given by 'regex'.
+ :param filename: The file path.
+ :param regex: The pattern to search for.
+ :returns: Whether the pattern was matched.
"""
with open(filename, encoding="utf-8") as file:
try:
@@ -258,9 +429,10 @@ def file_contains(filename, regex):
return False
-def get_gitlab_default():
+def get_gitlab_default() -> str:
"""
Get the name of the default branch on gitlab.
+ :returns: The branch name.
"""
repo_name = get_upstream_details().get("repo-name", None)
if repo_name is None:
@@ -283,12 +455,14 @@ def get_gitlab_default():
)
with urllib.request.urlopen(gitlab_request, timeout=20) as response:
- return json.load(response)["data"]["project"]["repository"]["rootRef"]
+ default = json.load(response)["data"]["project"]["repository"]["rootRef"]
+ assert isinstance(default, str)
+ return default
-def within_browser_root():
+def within_browser_root() -> bool:
"""
- Whether we are with the tor browser root.
+ :returns: Whether we are with the tor browser root.
"""
root = get_local_root()
if not root:
@@ -301,24 +475,24 @@ def within_browser_root():
# * -------------------- *
-def show_firefox_commit(_args):
+def show_firefox_commit(_args: argparse.Namespace) -> None:
"""
Print the tag name and commit for the last firefox commit below the current
HEAD.
"""
ref = get_firefox_ref("HEAD")
- print(ref.name)
+ print(ref.full_name)
print(ref.commit)
-def show_upstream_basis_commit(_args):
+def show_upstream_basis_commit(_args: argparse.Namespace) -> None:
"""
Print the last upstream commit for the current HEAD.
"""
print(get_upstream_basis_commit("HEAD"))
-def show_log(args):
+def show_log(args: argparse.Namespace) -> None:
"""
Show the git log between the current HEAD and the last firefox commit.
"""
@@ -326,7 +500,7 @@ def show_log(args):
git_run(["log", f"{commit}..HEAD", *args.gitargs], check=False)
-def show_files_containing(args):
+def show_files_containing(args: argparse.Namespace) -> None:
"""
List all the files that that have been modified for tor browser, that also
contain a regular expression.
@@ -336,33 +510,32 @@ def show_files_containing(args):
except re.error as err:
raise TbDevException(f"{args.regex} is not a valid python regex") from err
- file_list = get_changed_files(get_firefox_ref("HEAD").commit)
-
- for filename in file_list:
- if not os.path.isfile(filename):
+ for file_change in get_changed_files(get_firefox_ref("HEAD").commit):
+ path = file_change.new_path
+ if not os.path.isfile(path):
# deleted ofile
continue
- if file_contains(filename, regex):
- print(filename)
+ if file_contains(path, regex):
+ print(path)
-def show_changed_files(_args):
+def show_changed_files(_args: argparse.Namespace) -> None:
"""
List all the files that have been modified relative to upstream.
"""
- for filename in get_changed_files(get_upstream_basis_commit("HEAD")):
- print(filename)
+ for file_change in get_changed_files(get_upstream_basis_commit("HEAD")):
+ print(file_change.new_path)
-def lint_changed_files(args):
+def lint_changed_files(args: argparse.Namespace) -> None:
"""
Lint all the files that have been modified relative to upstream.
"""
os.chdir(get_local_root())
file_list = [
- f
+ f.new_path
for f in get_changed_files(get_upstream_basis_commit("HEAD"))
- if os.path.isfile(f) # Not deleted
+ if os.path.isfile(f.new_path) # Not deleted
]
# We add --warnings since clang only reports whitespace issues as warnings.
subprocess.run(
@@ -371,10 +544,18 @@ def lint_changed_files(args):
)
-def prompt_user(prompt, convert):
+# TODO: replace with "prompt_user[T](..., T]) -> T" after python 3.12 is the
+# minimum mach version.
+T = TypeVar("T")
+
+
+def prompt_user(prompt: str, convert: Callable[[str], T]) -> T:
"""
- Ask the user for some input until the given converter returns without
- throwing a ValueError.
+ Ask the user for some input.
+ :param prompt: The prompt to show the user.
+ :param convert: A method to convert the response into a type. Should
+ throw `ValueError` if the user should be re-prompted for a valid input.
+ :returns: The first valid user response.
"""
while True:
# Flush out stdin.
@@ -388,8 +569,12 @@ def prompt_user(prompt, convert):
pass
-def binary_reply_default_no(value):
- """Process a 'y' or 'n' reply, defaulting to 'n' if empty."""
+def binary_reply_default_no(value: str) -> bool:
+ """
+ Process a 'y' or 'n' reply, defaulting to 'n' if empty.
+ :param value: The user input.
+ :returns: Whether the answer is yes.
+ """
if value == "":
return False
if value.lower() == "y":
@@ -399,121 +584,737 @@ def binary_reply_default_no(value):
raise ValueError()
-def get_fixup_for_file(filename, firefox_commit):
- """Find the commit the given file should fix up."""
+class FixupTarget:
+ """Represents a commit that can be targeted by a fixup."""
+
+ def __init__(self, commit: str, short_ref: str, title: str) -> None:
+ """
+ :param commit: The commit hash for the commit.
+ :param short_ref: The shortened commit hash for display.
+ :param title: The first line of the commit message.
+ """
+ self.commit = commit
+ self.short_ref = short_ref
+ self.title = title
+ self.changes: list[FileChange] = []
+ self.fixups: list[FixupTarget] = []
+ self.target: None | FixupTarget = None
+
+ _FIXUP_REGEX = re.compile(r"^fixup! +")
+
+ def trim_fixup(self) -> tuple[str, int]:
+ """
+ Trim the "fixup!" prefixes.
+ :returns: The stripped commit title and the fixup depth (how many fixups
+ prefixes there were).
+ """
+ title = self.title
+ depth = 0
+ while True:
+ match = self.__class__._FIXUP_REGEX.match(title)
+ if not match:
+ return title, depth
+ title = title[match.end() :]
+ depth += 1
+
+ def touches_path(
+ self, path: str, filter_status: None | str = None, check_dir: bool = False
+ ) -> bool:
+ """
+ Whether this target, or one of its fixups or target, touches the given
+ path.
+ :param path: The path to check.
+ :param filter_status: Limit the detected changes to the given status(es).
+ :param check_dir: Whether we should treat `path` as a directory and check for
+ files within it.
+ :returns: Whether this target matches.
+ """
+ # NOTE: In the case of renames, we generally assume that renames occur
+ # in the fixup targets. E.g. "Commit 1" creates the file "file.txt", and
+ # "fixup! Commit 1" renames it to "new.txt". In this case, if the
+ # FixupTarget for "Commit 1" is passed in "file.txt" it will match. And
+ # if it is passed in "new.txt" it will also match via the self.fixups
+ # field, which will include the "fixup! Commit 1" rename.
+ # But the "fixup ! Commit 1" FixupTargets will only match with
+ # "file.txt" if they occurred before the rename fixup, and will only
+ # match with "new.txt" if they occur after the rename fixup. With the
+ # exception of the rename fixup itself, which will match both.
+ #
+ # In principle, we could identify a file across renames (have a mapping
+ # from each commit to what the file is called at that stage) and match
+ # using this file identifier. Similar to the "--follow" git diff
+ # argument. This would then cover cases where a rename occurs between
+ # the commit and its fixups, and allow fixups before the rename to also
+ # match. However, the former case is unexpected and the latter case
+ # would not be that useful.
+ if self._touches_path_basis(path, filter_status, check_dir):
+ return True
+ # Mark this as a valid target for the path if one of our fixups changes
+ # this path.
+ # NOTE: We use _touch_path_basis to prevent recursion. This means we
+ # will only check one layer up or down, but we only expect fixups of
+ # up to depth 1.
+ for fixup_target in self.fixups:
+ if fixup_target._touches_path_basis(path, filter_status, check_dir):
+ return True
+ # Mark this as a valid target if our target changes this path.
+ if self.target is not None and self.target._touches_path_basis(
+ path, filter_status, check_dir
+ ):
+ return True
+ return False
+
+ def _touches_path_basis(
+ self, path: str, filter_status: None | str, check_dir: bool
+ ) -> bool:
+ """
+ Whether this target touches the given path.
+ :param path: The path to check.
+ :param filter_status: Limit the detected changes to the given status.
+ :param check_dir: Whether we should treat `path` as a directory and check for
+ files within it.
+ :returns: Whether this target matches.
+ """
+ for file_change in self.changes:
+ if filter_status is not None and file_change.status not in filter_status:
+ continue
+ for test_path in (file_change.path, file_change.new_path):
+ if check_dir:
+ if os.path.commonpath((os.path.dirname(test_path), path)) == path:
+ # test_path's directory matches the path or is within it.
+ return True
+ elif test_path == path:
+ return True
+ return False
+
+
+def get_fixup_targets(
+ target_list: list[FixupTarget],
+ from_commit: str,
+ to_commit: str,
+ fixup_depth: int = 0,
+) -> None:
+ """
+ Find all the commits that can be targeted by a fixup between the given
+ commits.
+ :param target_list: The list to fill with targets. Appended in the order of
+ `from_commit` to `to_commit`.
+ :param from_commit: The commit to start from (non-inclusive).
+ :param to_commit: The commit to end on (inclusive).
+ :param fixup_depth: The maximum "depth" of fixups. I.e. how many "fixup!"
+ prefixes to allow.
+ """
+ raw_output = git_get(
+ [
+ "log",
+ "--pretty=format:%H,%h,%s",
+ "--reverse",
+ "--raw",
+ "-z",
+ f"{from_commit}..{to_commit}",
+ ],
+ strip=False,
+ )
+ pretty_regex = re.compile(
+ r"(?P<commit>[0-9a-f]+),(?P<short_ref>[0-9a-f]+),(?P<title>[^\n\0]*)\n"
+ )
+ excluded_regex_list = [
+ re.compile(r"^Bug [0-9]+.*r="), # Backported Mozilla bug.
+ re.compile(r"^dropme! "),
+ ]
+
+ while raw_output:
+ match = pretty_regex.match(raw_output)
+ if not match:
+ raise ValueError(f"Invalid pretty format: {raw_output[:100]}...")
+ fixup_target = FixupTarget(
+ match.group("commit"), match.group("short_ref"), match.group("title")
+ )
+ raw_output = raw_output[match.end() :]
+ while raw_output and raw_output[0] != "\0":
+ file_change, end = parse_raw_diff_line(raw_output)
+ fixup_target.changes.append(file_change)
+ raw_output = raw_output[end:]
+ if raw_output:
+ # Skip over the "\0".
+ raw_output = raw_output[1:]
+
+ for regex in excluded_regex_list:
+ if regex.match(fixup_target.title):
+ # Exclude from the list.
+ continue
+
+ trimmed_title, depth = fixup_target.trim_fixup()
+ if depth:
+ original_target = None
+ for target in target_list:
+ if target.title == trimmed_title:
+ original_target = target
+ break
+
+ if original_target:
+ original_target.fixups.append(fixup_target)
+ fixup_target.target = original_target
+ if depth > fixup_depth:
+ # Exclude from the list.
+ continue
+
+ target_list.append(fixup_target)
+
+
+class NewCommitBasis:
+ def __init__(self) -> None:
+ self.staged_paths: set[str] = set()
+ self.adding_paths: set[str] = set()
+
+ def add(self, paths: Iterable[str], staged: bool) -> None:
+ """
+ Add a path to include in this commit.
+ :param paths: The paths to add.
+ :param staged: Whether we are adding already staged changes.
+ """
+ if staged:
+ self.staged_paths.update(paths)
+ return
+
+ self.adding_paths.update(paths)
+
+
+class NewCommit(NewCommitBasis):
+ """Represents a new commit that we want to create."""
- def parse_log_line(line):
- [commit, short_ref, title] = line.split(",", 2)
- return {"commit": commit, "short-ref": short_ref, "title": title}
+ def __init__(self, alias: str) -> None:
+ """
+ :param alias: The alias name for the commit.
+ """
+ super().__init__()
+ self.alias = alias
- options = [
- parse_log_line(line)
- for line in git_get(
- [
- "log",
- "--pretty=format:%H,%h,%s",
- f"{firefox_commit}..HEAD",
- "--",
- filename,
- ]
+
+class NewFixup(NewCommitBasis):
+ """Represents a new fixup commit that we want to create."""
+
+ def __init__(self, target: FixupTarget) -> None:
+ """
+ :param target: The commit to target with the fixup.
+ """
+ super().__init__()
+ self.target = target
+
+
+def get_suggested_fixup_targets_for_change(
+ file_change: FileChange,
+ fixup_target_list: list[FixupTarget],
+ firefox_directories_lazy: Callable[[], set[str]],
+) -> Iterator[FixupTarget]:
+ """
+ Find the suggested fixup targets for the given file change.
+ :param file_change: The file change to get a suggestion for.
+ :param fixup_target_list: The list to choose from.
+ :param firefox_directories_lazy: Lazy method to return the firefox
+ directories.
+ :yield: The suggested fixup targets.
+ """
+
+ def filter_list(
+ path: str, filter_status: None | str = None, check_dir: bool = False
+ ) -> Iterator[FixupTarget]:
+ return (
+ t
+ for t in fixup_target_list
+ if t.touches_path(path, filter_status=filter_status, check_dir=check_dir)
)
+
+ if file_change.status == "D":
+ # Deleted.
+ # Find the commit that introduced this file or previously deleted it.
+ # I.e. added the file ("A"), renamed it ("R"), or deleted it ("D").
+ yield from filter_list(file_change.path, filter_status="ARD")
+ return
+
+ if file_change.status == "A":
+ # First check to see if this file name was actually touched before.
+ yielded_target = False
+ for target in filter_list(file_change.path):
+ yielded_target = True
+ yield target
+ if yielded_target:
+ return
+ # Else, find commits that introduced files in the same directory, or
+ # deleted in them, if they are not firefox directories.
+ dir_path = file_change.path
+ while True:
+ dir_path = os.path.dirname(dir_path)
+ if not dir_path or dir_path in firefox_directories_lazy():
+ return
+
+ yielded_target = False
+ for target in filter_list(dir_path, filter_status="ARD", check_dir=True):
+ yielded_target = True
+ yield target
+
+ if yielded_target:
+ return
+ # Else, search one directory higher.
+
+ if file_change.status == "R":
+ # Renamed.
+ # Find the commit that introduced the original name for this file.
+ yield from filter_list(file_change.path, filter_status="AR")
+ return
+
+ # Modified.
+ yield from filter_list(file_change.path)
+
+
+def ask_for_target(
+ file_change_list: list[FileChange],
+ new_commits_list: list[NewCommit | NewFixup],
+ suggested_fixup_target_list: list[FixupTarget],
+ full_fixup_target_list: list[FixupTarget],
+ staged: bool = False,
+) -> bool:
+ """
+ Ask the user to choose a target.
+ :param file_change_list: The file changes to ask for.
+ :param new_commits_list: The list of pending new commits, may be added to.
+ :param suggested_fixup_target_list: The list of suggested target fixups
+ to choose from.
+ :param staged: Whether this is for staged changes.
+ :returns: `True` if the operation should be aborted.
+ """
+
+ new_paths = [c.new_path for c in file_change_list]
+ all_paths = set(new_paths).union(c.path for c in file_change_list)
+ non_fixup_commits: list[NewCommit] = [
+ n for n in new_commits_list if isinstance(n, NewCommit)
]
- if not options:
- print(f"No commit found for {filename}")
- return None
- def valid_index(val):
+ shown_list: list[NewCommit | FixupTarget] = (
+ non_fixup_commits + suggested_fixup_target_list
+ )
+
+ can_skip = not staged
+ shown_full = False
+
+ index_offset = 2
+
+ def valid_response(val: str) -> tuple[str, None | NewCommit | FixupTarget]:
+ val = val.strip()
+
+ if val == "h":
+ return "help", None
+
+ if val == "a":
+ return "abort", None
+
if val == "d":
- return val
+ return "diff", None
+
+ if val == "f":
+ if shown_full:
+ # Already done once.
+ raise ValueError()
+ return "full-list", None
+ is_patch_full = val.startswith("P")
is_patch = val.startswith("p")
- if is_patch:
- val = val[1:]
+ if is_patch or is_patch_full:
+ index = int(val[1:], base=10) # Raises ValueError if not integer.
+ else:
+ index = int(val, base=10) # Raises ValueError if not integer.
+ if index == 0:
+ if not can_skip:
+ raise ValueError()
+ return "skip", None
+
+ if index == 1:
+ return "new", None
- # May raise a ValueError.
- as_index = int(val)
- if as_index < 0 or as_index > len(options):
+ index -= index_offset
+
+ if index < 0 or index >= len(shown_list):
raise ValueError()
- if as_index == 0:
- if is_patch:
+ selected = shown_list[index]
+
+ if is_patch_full:
+ return "patch-full", selected
+ if is_patch:
+ return "patch", selected
+ return "target", selected
+
+ def alias_response(val: str) -> str:
+ # Choose a default alias name if none is given.
+ val = val.strip() or f"New commit {len(non_fixup_commits)}"
+ for new_commit in non_fixup_commits:
+ if new_commit.alias == val:
+ # Already in use.
raise ValueError()
- return None
+ return val
+
+ def print_index_option(index: int, description: str) -> None:
+ print(f" \x1b[1m{index}\x1b[0m: {description}")
- return (is_patch, options[as_index - 1]["commit"])
+ def in_pink(text: str) -> str:
+ return f"\x1b[1;38;5;212m{text}\x1b[0m"
+ prefix_str = "For " + (in_pink("staged") if staged else "unstaged") + " changes to"
+ if len(new_paths) == 1:
+ print(f"{prefix_str} {in_pink(new_paths[0])}:")
+ else:
+ print(f"{prefix_str}:")
+ for path in new_paths:
+ print(f" {in_pink(path)}")
+ print("")
+
+ show_help = True
+ reshow_list = True
while True:
- print(f"For {filename}:\n")
- print(" \x1b[1m0\x1b[0m: None")
- for index, opt in enumerate(options):
- print(
- f" \x1b[1m{index + 1}\x1b[0m: "
- + f"\x1b[1;38;5;212m{opt['short-ref']}\x1b[0m "
- + opt["title"]
- )
+ if reshow_list:
+ if can_skip:
+ print_index_option(0, "Skip")
+ print_index_option(1, "New commit")
+ for index, target in enumerate(shown_list, start=index_offset):
+ if isinstance(target, NewCommit):
+ print_index_option(index, f"Add to new commit: {target.alias}")
+ else:
+ print_index_option(
+ index, f"Fixup: {in_pink(target.short_ref)} {target.title}"
+ )
+ reshow_list = False
print("")
- response = prompt_user(
- "Choose an <index> to fixup, or '0' to skip this file, "
- "or 'd' to view the pending diff, "
- "or 'p<index>' to view the patch for the index: ",
- valid_index,
+
+ response, selected = prompt_user(
+ (
+ "Choose an <index> to target. Type 'h' for additional options: "
+ if show_help
+ else "Choose an <index> to target or an option: "
+ ),
+ valid_response,
)
- if response is None:
- # Skip this file.
- return None
- if response == "d":
- git_run(["diff", "--", filename])
+ if response == "help":
+ print("Options:")
+ for option, desc in (
+ ("h", "show the available options."),
+ ("a", "abort this commit operation and all pending commits."),
+ (
+ ("", "")
+ if shown_full
+ else (
+ "f",
+ "show the full list of fixup targets, rather than just the suggested ones.",
+ )
+ ),
+ ("d", "view the diff for the pending file changes."),
+ (
+ "P<index>",
+ "view the patch for the index (including its relevant fixups).",
+ ),
+ (
+ "p<index>",
+ "view the patch for the index (including its relevant fixups), "
+ "limited to the current files.",
+ ),
+ ):
+ if not option:
+ # Skip this option.
+ continue
+ print(f" \x1b[1m{option[0]}\x1b[0m{option[1:].ljust(7)}: {desc}")
+ # Do not show the help option again.
+ show_help = False
+ continue
+
+ if response == "abort":
+ return True
+
+ if response == "skip":
+ return False
+
+ if response == "new":
+ new_alias = prompt_user(
+ "Enter an optional temporary alias for this new commit: ",
+ alias_response,
+ )
+ new_commit = NewCommit(new_alias)
+ new_commit.add(all_paths, staged)
+ new_commits_list.append(new_commit)
+ return False
+
+ if response == "target":
+ assert selected is not None
+
+ if isinstance(selected, NewCommit):
+ # Adding to a new commit.
+ selected.add(all_paths, staged)
+ return False
+
+ for new_fixup in new_commits_list:
+ if not isinstance(new_fixup, NewFixup):
+ continue
+ if new_fixup.target == selected:
+ # We already have a pending fixup commit that targets this
+ # selected target. Add this path to the same commit.
+ new_fixup.add(all_paths, staged)
+ return False
+
+ new_fixup = NewFixup(selected)
+ new_fixup.add(all_paths, staged)
+ new_commits_list.append(new_fixup)
+ return False
+
+ if response == "full-list":
+ shown_list = non_fixup_commits + full_fixup_target_list
+ shown_full = True
+ reshow_list = True
continue
- view_patch, commit = response
- if view_patch:
- git_run(["log", "-p", "-1", commit, "--", filename])
+ if response == "diff":
+ git_args = ["diff", "--color"]
+ if staged:
+ git_args.append("--staged")
+ git_args.extend(git_path_args(all_paths))
+ git_run_pager(git_args)
continue
- return commit
+ if response in ("patch", "patch-full"):
+ assert selected is not None
+
+ filter_paths = response == "patch"
+
+ if isinstance(selected, NewCommit):
+ git_sequence = [
+ ["diff", "--color", "--staged", *git_path_args((path,))]
+ for path in selected.staged_paths
+ if not filter_paths or path in all_paths
+ ]
+ git_sequence.extend(
+ ["diff", "--color", *git_path_args((path,))]
+ for path in selected.adding_paths
+ if not filter_paths or path in all_paths
+ )
+
+ # Show what the expected patch will be for the new commit.
+ git_run_pager(
+ arg_sequence=git_sequence, pager_prefix=f"{selected.alias}\n\n"
+ )
+ else:
+ # Show the log entry for the FixupTarget and each of its fixups.
+ # Order with the commmit closest to HEAD first. We expect
+ # selected.fixups to match this order.
+ git_sequence = []
+ # If `filter_paths` is set, we want to limit the log to the
+ # paths, and try to track any renames in the commit history.
+ prev_log_paths: None | set[str] = None
+ # For the first commit in the sequence, we use the old path
+ # names (rather than `c.new_path`) since we expect the commit
+ # which is closest to us to use the older names.
+ log_paths: None | set[str] = (
+ {c.path for c in file_change_list} if filter_paths else None
+ )
+ for target in (*selected.fixups, selected):
+ git_args = [
+ "log",
+ "--color",
+ "-p",
+ f"{target.commit}~1..{target.commit}",
+ ]
+ if filter_paths:
+ assert log_paths is not None
+ # Track the renamed paths.
+ prev_log_paths = log_paths.copy()
+ for file_change in target.changes:
+ if (
+ file_change.status == "R"
+ and file_change.new_path in log_paths
+ ):
+ # file was renamed in this change.
+ # Update log_paths to the new name.
+ # NOTE: This should have a similar effect to the
+ # --follow option for git log for a single file
+ # NOTE: File renames will not be properly
+ # tracked if a rename occurs outside of
+ # `selected.changes` or
+ # `selected.fixups[].changes`, but this is
+ # unexpected.
+ log_paths.remove(file_change.new_path)
+ log_paths.add(file_change.path)
+
+ # NOTE: This log entry may be empty if none of the paths
+ # match.
+ # NOTE: We include both log_paths and prev_log_paths to
+ # show renames in the diff output.
+ git_args.extend(git_path_args(log_paths | prev_log_paths))
+ git_sequence.append(git_args)
+ # Combine all the logs into one.
+ git_run_pager(arg_sequence=git_sequence)
+ continue
+
+ raise ValueError(f"Unexpected response: {response}")
-def auto_fixup(_args):
+def auto_commit(_args: argparse.Namespace) -> None:
"""
- Automatically find and fix up commits using the current unstaged changes.
+ Automatically find and fix up commits for any pending changes.
"""
+ # Want git log and add to be run from the root.
+ os.chdir(get_local_root())
# Only want to search as far back as the firefox commit.
firefox_commit = get_firefox_ref("HEAD").commit
- staged_files = get_changed_files("HEAD", staged=True)
- if staged_files:
- raise TbDevException(f"Have already staged files: {staged_files}")
+ staged_changes = [f for f in get_changed_files(staged=True)]
+ if staged_changes:
+ print("Existing staged changes for:")
+ for file_change in staged_changes:
+ print(f" {file_change.new_path}")
+ if not prompt_user(
+ "Include staged changes? (y/\x1b[4mn\x1b[0m)", binary_reply_default_no
+ ):
+ raise TbDevException("Cannot continue with pending staged changes")
+ print("")
- fixups = {}
- for filename in get_changed_files("HEAD"):
- commit = get_fixup_for_file(filename, firefox_commit)
- if commit is None:
+ full_target_list: list[FixupTarget] = []
+ # Determine if HEAD points to a branch or not and has an upstream commit.
+ # We choose check=False since the exit status is non-zero when we are in a
+ # detached state.
+ head_symbolic_ref = git_get(["symbolic-ref", "-q", "HEAD"], check=False)
+ if not head_symbolic_ref or not bool(
+ git_get(["for-each-ref", "--format=%(upstream)", head_symbolic_ref])
+ ):
+ # Unexpected, but not fatal.
+ print("HEAD has no upstream tracking!")
+ # Just include all commits since firefox_commit with no fixup depth
+ get_fixup_targets(full_target_list, firefox_commit, "HEAD", fixup_depth=0)
+ else:
+ upstream_commit = get_upstream_basis_commit("HEAD")
+ # Only include "fixup!" commits that are between here and the upstream
+ # tracking commit.
+ get_fixup_targets(
+ full_target_list, firefox_commit, upstream_commit, fixup_depth=0
+ )
+ get_fixup_targets(full_target_list, upstream_commit, "HEAD", fixup_depth=1)
+
+ # full_target_list is ordered with the earlier commits first. Reverse this.
+ full_target_list.reverse()
+ # Also reverse the fixups order to follow the same order.
+ for target in full_target_list:
+ target.fixups.reverse()
+
+ # Lazy load the list of firefox directories since they are unlikely to be
+ # needed.
+ @functools.cache
+ def firefox_directories_lazy() -> set[str]:
+ return {
+ dir_name
+ for dir_name in git_get(
+ [
+ "ls-tree",
+ "-r",
+ "-d",
+ "--name-only",
+ "--full-tree",
+ "-z",
+ firefox_commit,
+ ],
+ strip=False,
+ ).split("\0")
+ if dir_name
+ }
+
+ # Check untracked files to be added.
+ for path in git_get(
+ ["ls-files", "--other", "--exclude-standard", "-z"], strip=False
+ ).split("\0"):
+ if not path:
continue
- if commit not in fixups:
- fixups[commit] = [filename]
- else:
- fixups[commit].append(filename)
+ if prompt_user(
+ f"Start tracking file `{path}`? (y/\x1b[4mn\x1b[0m)",
+ binary_reply_default_no,
+ ):
+ # Include in the git diff output, but do not stage.
+ git_run(["add", "--intent-to-add", path])
print("")
- for commit, files in fixups.items():
- print("")
- git_run(["add", *files])
- git_run(["commit", f"--fixup={commit}"])
+ aborted = False
+ new_commits_list: list[NewCommit | NewFixup] = []
+ # First go through staged changes.
+ if staged_changes:
+ common_fixup_targets = None
+ for change in staged_changes:
+ target_iter = get_suggested_fixup_targets_for_change(
+ change, full_target_list, firefox_directories_lazy
+ )
+ if common_fixup_targets is None:
+ common_fixup_targets = set(target_iter)
+ else:
+ common_fixup_targets.intersection_update(target_iter)
+
+ assert common_fixup_targets is not None
+
+ aborted = ask_for_target(
+ staged_changes,
+ new_commits_list,
+ # Sort in the same order as full_target_list.
+ [target for target in full_target_list if target in common_fixup_targets],
+ full_target_list,
+ staged=True,
+ )
print("")
- if prompt_user(
- "Edit fixup commit message? (y/\x1b[4mn\x1b[0m)", binary_reply_default_no
- ):
+ if not aborted:
+ for file_change in get_changed_files():
+ target_list = list(
+ get_suggested_fixup_targets_for_change(
+ file_change, full_target_list, firefox_directories_lazy
+ )
+ )
+ aborted = ask_for_target(
+ [file_change],
+ new_commits_list,
+ target_list,
+ full_target_list,
+ staged=False,
+ )
+ print("")
+ if aborted:
+ break
+
+ if aborted:
+ return
+
+ # NOTE: Only the first commit can include staged changes.
+ # This should already be the case, but we want to double check.
+ for commit_index in range(1, len(new_commits_list)):
+ if new_commits_list[commit_index].staged_paths:
+ raise ValueError(f"Staged changes for commit {commit_index}")
+
+ for new_commit in new_commits_list:
+ print("")
+ if new_commit.adding_paths:
+ git_run(["add", *git_path_args(new_commit.adding_paths)])
+ if isinstance(new_commit, NewFixup):
+ git_run(["commit", f"--fixup={new_commit.target.commit}"])
+ print("")
+ is_double_fixup = bool(new_commit.target.target)
+ if not is_double_fixup and prompt_user(
+ "Edit fixup commit message? (y/\x1b[4mn\x1b[0m)",
+ binary_reply_default_no,
+ ):
+ git_run(["commit", "--amend"])
+ print("")
+ else:
+ git_run(["commit", "-m", new_commit.alias])
git_run(["commit", "--amend"])
+ print("")
-def clean_fixups(_args):
+def clean_fixups(_args: argparse.Namespace) -> None:
"""
Perform an interactive rebase that automatically applies fixups, similar to
--autosquash but also works on fixups of fixups.
"""
- user_editor = git_get(["var", "GIT_SEQUENCE_EDITOR"])[0]
+ user_editor = git_get(["var", "GIT_SEQUENCE_EDITOR"])
sub_editor = os.path.join(
os.path.dirname(os.path.realpath(__file__)), FIXUP_PREPROCESSOR_EDITOR
)
@@ -525,7 +1326,7 @@ def clean_fixups(_args):
)
-def show_default(_args):
+def show_default(_args: argparse.Namespace) -> None:
"""
Print the default branch name from gitlab.
"""
@@ -536,7 +1337,7 @@ def show_default(_args):
print(f"{upstream}/{default_branch}")
-def branch_from_default(args):
+def branch_from_default(args: argparse.Namespace) -> None:
"""
Fetch the default gitlab branch from upstream and create a new local branch.
"""
@@ -557,7 +1358,7 @@ def branch_from_default(args):
)
-def move_to_default(args):
+def move_to_default(args: argparse.Namespace) -> None:
"""
Fetch the default gitlab branch from upstream and move the specified
branch's commits on top. A new branch will be created tracking the default
@@ -569,7 +1370,7 @@ def move_to_default(args):
if branch_name is None:
# Use current branch as default.
try:
- branch_name = git_get(["branch", "--show-current"])[0]
+ branch_name = git_get(["branch", "--show-current"])
except IndexError:
raise TbDevException("No current branch")
@@ -608,7 +1409,7 @@ def move_to_default(args):
git_run(["cherry-pick", f"{current_basis}..{old_branch_name}"], check=False)
-def show_range_diff(args):
+def show_range_diff(args: argparse.Namespace) -> None:
"""
Show the range diff between two branches, from their firefox bases.
"""
@@ -624,21 +1425,21 @@ def show_range_diff(args):
)
-def show_diff_diff(args):
+def show_diff_diff(args: argparse.Namespace) -> None:
"""
Show the diff between the diffs of two branches, relative to their firefox
bases.
"""
- config_res = git_get(["config", "--get", "diff.tool"])
- if not config_res:
+ try:
+ diff_tool = next(git_lines(["config", "--get", "diff.tool"]))
+ except StopIteration:
raise TbDevException("No diff.tool configured for git")
- diff_tool = config_res[0]
# Filter out parts of the diff we expect to be different.
index_regex = re.compile(r"index [0-9a-f]{12}\.\.[0-9a-f]{12}")
lines_regex = re.compile(r"@@ -[0-9]+,[0-9]+ \+[0-9]+,[0-9]+ @@(?P<rest>.*)")
- def save_diff(branch):
+ def save_diff(branch: str) -> str:
firefox_commit = get_firefox_ref(branch).commit
file_desc, file_name = tempfile.mkstemp(
text=True, prefix=f'{branch.split("/")[-1]}-'
@@ -653,6 +1454,7 @@ def show_diff_diff(args):
)
with os.fdopen(file_desc, "w") as file:
+ assert diff_process.stdout is not None
for line in diff_process.stdout:
if index_regex.match(line):
# Fake data that will match.
@@ -665,7 +1467,7 @@ def show_diff_diff(args):
continue
file.write(line)
- status = diff_process.poll()
+ status = diff_process.wait()
if status != 0:
raise TbDevException(f"git diff exited with status {status}")
@@ -681,7 +1483,7 @@ def show_diff_diff(args):
# * -------------------- *
-def branch_complete(prefix, parsed_args, **kwargs):
+def branch_complete(prefix: str, **_kwargs: Any) -> list[str]:
"""
Complete the argument with a branch name.
"""
@@ -689,7 +1491,7 @@ def branch_complete(prefix, parsed_args, **kwargs):
return []
try:
branches = [ref.name for ref in get_refs("head", "")]
- branches.extend([ref.name for ref in get_refs("remote", "")])
+ branches.extend(ref.name for ref in get_refs("remote", ""))
branches.append("HEAD")
except Exception:
return []
@@ -699,7 +1501,20 @@ def branch_complete(prefix, parsed_args, **kwargs):
parser = argparse.ArgumentParser()
subparsers = parser.add_subparsers(required=True)
-for name, details in {
+
+class ArgConfig(TypedDict):
+ help: str
+ metavar: NotRequired[str]
+ nargs: NotRequired[str]
+ completer: NotRequired[Callable[[str], list[str]]]
+
+
+class CommandConfig(TypedDict):
+ func: Callable[[argparse.Namespace], None]
+ args: NotRequired[dict[str, ArgConfig]]
+
+
+all_commands: dict[str, CommandConfig] = {
"show-upstream-basis-commit": {
"func": show_upstream_basis_commit,
},
@@ -716,8 +1531,8 @@ for name, details in {
},
},
},
- "auto-fixup": {
- "func": auto_fixup,
+ "auto-commit": {
+ "func": auto_commit,
},
"clean-fixups": {
"func": clean_fixups,
@@ -794,20 +1609,25 @@ for name, details in {
"regex": {"help": "the regex that the files must contain"},
},
},
-}.items():
- help_message = re.sub(r"\s+", " ", details["func"].__doc__).strip()
+}
+
+for name, command_config in all_commands.items():
+ help_message = command_config["func"].__doc__
+ assert isinstance(help_message, str)
+ help_message = re.sub(r"\s+", " ", help_message).strip()
sub = subparsers.add_parser(name, help=help_message)
- sub.set_defaults(func=details["func"])
- for arg, keywords in details.get("args", {}).items():
+ sub.set_defaults(func=command_config["func"])
+ for arg, keywords in command_config.get("args", {}).items():
completer = None
if "completer" in keywords:
completer = keywords["completer"]
del keywords["completer"]
sub_arg = sub.add_argument(arg, **keywords)
- if completer is not None:
- sub_arg.completer = completer
+ if completer is not None and argcomplete is not None:
+ sub_arg.completer = completer # type: ignore
-argcomplete.autocomplete(parser)
+if argcomplete is not None:
+ argcomplete.autocomplete(parser)
try:
if not within_browser_root():
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/f12904…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/f12904…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/tor-browser][tor-browser-146.0a1-16.0-2] 7 commits: fixup! BB 41803: Add some developer tools for working on tor-browser.
by henry (ï¼ henry) 11 Dec '25
by henry (ï¼ henry) 11 Dec '25
11 Dec '25
henry pushed to branch tor-browser-146.0a1-16.0-2 at The Tor Project / Applications / Tor Browser
Commits:
25553432 by Henry Wilkes at 2025-12-11T13:53:13+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Make git_get return the stdout string, rather than a list.
Add git_lines to generate lines.
- - - - -
523bc569 by Henry Wilkes at 2025-12-11T13:53:15+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Use raw diff to get list of file changes.
- - - - -
20c68b54 by Henry Wilkes at 2025-12-11T13:53:16+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Add type annotations and parameter documentation.
- - - - -
a0bff3f5 by Henry Wilkes at 2025-12-11T13:53:17+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Make the argcomplete module optional.
- - - - -
a64f9e20 by Henry Wilkes at 2025-12-11T13:53:18+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Use function caching instead of global variables.
- - - - -
80acc6a0 by Henry Wilkes at 2025-12-11T13:53:19+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Fetch FIREFOX_ tags from the remote if they are missing.
- - - - -
1b34bbe4 by Henry Wilkes at 2025-12-11T13:53:21+00:00
fixup! BB 41803: Add some developer tools for working on tor-browser.
TB 44367: Improve the auto-fixup/auto-commit command.
The auto-fixup command was renamed to auto-commit. It now also handles:
1. Already staged changes.
2. Untracked/added files.
3. Removed files.
4. Renamed files.
5. Allowing the user to create a new commit.
- - - - -
1 changed file:
- tools/base_browser/tb-dev
Changes:
=====================================
tools/base_browser/tb-dev
=====================================
@@ -6,6 +6,7 @@ Useful tools for working on tor-browser repository.
import argparse
import atexit
+import functools
import json
import os
import re
@@ -14,8 +15,15 @@ import sys
import tempfile
import termios
import urllib.request
+from collections.abc import Callable, Iterable, Iterator
+from types import ModuleType
+from typing import Any, NotRequired, TypedDict, TypeVar
-import argcomplete
+argcomplete: None | ModuleType = None
+try:
+ import argcomplete
+except ImportError:
+ pass
GIT_PATH = "/usr/bin/git"
UPSTREAM_URLS = {
@@ -36,9 +44,14 @@ class TbDevException(Exception):
pass
-def git_run(args, check=True, env=None):
+def git_run(
+ args: list[str], check: bool = True, env: None | dict[str, str] = None
+) -> None:
"""
Run a git command with output sent to stdout.
+ :param args: The arguments to pass to git.
+ :param check: Whether to check for success.
+ :param env: Optional environment to set.
"""
if env is not None:
tmp_env = dict(os.environ)
@@ -51,46 +64,122 @@ def git_run(args, check=True, env=None):
raise TbDevException(str(err)) from err
-def git_get(args):
+def git_run_pager(
+ args: list[str] | None = None,
+ arg_sequence: Iterable[list[str]] | None = None,
+ pager_prefix: None | str = None,
+) -> None:
"""
- Run a git command with each non-empty line returned in a list.
+ Run a sequence of git commands with the output concatenated and sent to the
+ git pager.
+ :param args: The arguments to pass to git, or `None` if a sequence is desired.
+ :param arg_sequence: A sequence representing several git commands.
+ :param pager_prefix: An optional text to send to the pager first.
+ """
+ if arg_sequence is None:
+ if args is not None:
+ arg_sequence = (args,)
+ else:
+ raise ValueError("Missing `arg_sequence` or `args`")
+ elif args is not None:
+ raise ValueError("Unexpected both args and arg_sequence")
+
+ pager = git_get(["var", "GIT_PAGER"])
+ if not pager:
+ raise TbDevException("Missing a GIT_PAGER")
+ command = [pager]
+ if os.path.basename(pager) == "less":
+ # Show colours.
+ command.append("-R")
+
+ pager_process = subprocess.Popen(command, stdin=subprocess.PIPE, text=True)
+ assert pager_process.stdin is not None
+
+ if pager_prefix is not None:
+ pager_process.stdin.write(pager_prefix)
+ pager_process.stdin.flush()
+
+ for git_args in arg_sequence:
+ subprocess.run(
+ [GIT_PATH, "--no-pager", *git_args], check=False, stdout=pager_process.stdin
+ )
+
+ pager_process.stdin.close()
+
+ status = pager_process.wait()
+ if status != 0:
+ raise TbDevException(f"git pager {pager} exited with status {status}")
+
+
+def git_get(args: list[str], strip: bool = True, check: bool = True) -> str:
+ """
+ Return the output from a git command.
+ :param args: The arguments to send to git.
+ :param strip: Whether to strip the whitespace from the output.
+ :param check: Whether to check for success.
+ :returns: The stdout.
"""
try:
git_process = subprocess.run(
- [GIT_PATH, *args], text=True, stdout=subprocess.PIPE, check=True
+ [GIT_PATH, *args], text=True, stdout=subprocess.PIPE, check=check
)
except subprocess.CalledProcessError as err:
raise TbDevException(str(err)) from err
- return [line for line in git_process.stdout.split("\n") if line]
+ ret = git_process.stdout
+ if strip:
+ ret = ret.strip()
+ return ret
-local_root = None
+def git_lines(args: list[str]) -> Iterator[str]:
+ """
+ Yields the non-empty lines returned by the git command.
+ :param args: The arguments to send to git.
+ :yield: The lines.
+ """
+ for line in git_get(args, strip=False).split("\n"):
+ if not line:
+ continue
+ yield line
+
+
+def git_path_args(path_iter: Iterable[str]) -> Iterator[str]:
+ """
+ Generate the trailing arguments to specify paths in git commands, includes
+ the "--" separator just before the paths.
+ :param path_iter: The paths that should be passed in.
+ :yields: The git arguments.
+ """
+ yield "--"
+ for path in path_iter:
+ yield f":(literal){path}"
-def get_local_root():
+(a)functools.cache
+def get_local_root() -> str:
"""
Get the path for the tor-browser root directory.
+ :returns: The local root.
"""
- global local_root
- if local_root is None:
- try:
- # Make sure we have a matching remote in this git repository.
- if get_upstream_details()["is-browser-repo"]:
- local_root = git_get(["rev-parse", "--show-toplevel"])[0]
- else:
- local_root = ""
- except TbDevException:
- local_root = ""
- return local_root
+ try:
+ # Make sure we have a matching remote in this git repository.
+ if get_upstream_details()["is-browser-repo"] == "True":
+ return git_get(["rev-parse", "--show-toplevel"])
+ else:
+ return ""
+ except TbDevException:
+ return ""
-def determine_upstream_details():
+(a)functools.cache
+def get_upstream_details() -> dict[str, str]:
"""
- Determine details about the upstream.
+ Get details about the upstream repository.
+ :returns: The details.
"""
remote_urls = {
- remote: git_get(["remote", "get-url", remote])[0]
- for remote in git_get(["remote"])
+ remote: git_get(["remote", "get-url", remote])
+ for remote in git_lines(["remote"])
}
matches = {
@@ -102,7 +191,7 @@ def determine_upstream_details():
}
is_browser_repo = len(matches) > 0
- details = {"is-browser-repo": is_browser_repo}
+ details = {"is-browser-repo": str(is_browser_repo)}
origin_remote_repo = matches.get("origin", None)
upstream_remote_repo = matches.get("upstream", None)
@@ -125,31 +214,30 @@ def determine_upstream_details():
return details
-cached_upstream_details = None
-
-
-def get_upstream_details():
- """
- Get details about the upstream repository.
- """
- global cached_upstream_details
- if cached_upstream_details is None:
- cached_upstream_details = determine_upstream_details()
- return cached_upstream_details
-
-
class Reference:
"""Represents a git reference to a commit."""
- def __init__(self, name, commit):
- self.name = name
+ _REFS_REGEX = re.compile(r"refs/[a-z]+/")
+
+ def __init__(self, full_name: str, commit: str) -> None:
+ """
+ :param full_name: The full reference name. E.g. "refs/tags/MyTag".
+ :param commit: The commit hash for the commit this reference points to.
+ """
+ match = self.__class__._REFS_REGEX.match(full_name)
+ if not match:
+ raise ValueError(f"Invalid reference name {full_name}")
+ self.full_name = full_name
+ self.name = full_name[match.end() :]
self.commit = commit
-def get_refs(ref_type, name_start):
+def get_refs(ref_type: str, name_start: str) -> Iterator[Reference]:
"""
- Get a list of references that match the given 'ref_type' ("tag" or "remote"
- or "head") that starts with the given 'name_start'.
+ Get a list of references that match the given conditions.
+ :param ref_type: The ref type to search for ("tag" or "remote" or "head").
+ :param name_start: The ref name start to match against.
+ :yield: The matching references.
"""
if ref_type == "tag":
ref_start = "refs/tags/"
@@ -163,56 +251,83 @@ def get_refs(ref_type, name_start):
fstring = "%(*objectname),%(objectname),%(refname)"
pattern = f"{ref_start}{name_start}**"
- def line_to_ref(line):
+ def line_to_ref(line: str) -> Reference:
[objectname_reference, objectname, ref_name] = line.split(",", 2)
# For annotated tags, the objectname_reference is non-empty and points
# to an actual commit.
# For remotes, heads and lightweight tags, the objectname_reference will
# be empty and objectname will point directly to the commit.
- return Reference(
- ref_name.replace(ref_start, "", 1), objectname_reference or objectname
- )
+ return Reference(ref_name, objectname_reference or objectname)
- return [
+ return (
line_to_ref(line)
- for line in git_get(["for-each-ref", f"--format={fstring}", pattern])
- ]
+ for line in git_lines(["for-each-ref", f"--format={fstring}", pattern])
+ )
-def get_nearest_ref(ref_type, name_start, search_from):
+def get_firefox_ref(search_from: str) -> Reference:
"""
- Search backwards from the 'search_from' commit to find the first commit
- that matches the given 'ref_type' that starts with the given 'name_start'.
+ Search for the commit that comes from firefox.
+ :param search_from: The commit to search backwards from.
+ :returns: The firefox reference.
"""
- ref_list = get_refs(ref_type, name_start)
+ # Only search a limited history that should include the FIREFOX_ tag.
+ search_commits = [c for c in git_lines(["rev-list", "-1000", search_from])]
+
+ firefox_tag_prefix = "FIREFOX_"
- for commit in git_get(["rev-list", "-1000", search_from]):
- for ref in ref_list:
+ existing_tags = list(get_refs("tag", firefox_tag_prefix))
+ for commit in search_commits:
+ for ref in existing_tags:
if commit == ref.commit:
return ref
- raise TbDevException(f"No {name_start} commit found in the last 1000 commits")
-
-
-def get_firefox_ref(search_from):
+ # Might just need to fetch tags from the remote.
+ upstream = get_upstream_details().get("remote", None)
+ if upstream:
+ remote_ref: None | Reference = None
+ search_index = len(search_commits)
+ # Search the remote for a tag that is in our history.
+ # We want to avoid triggering a long fetch, so we just want to grab the
+ # tag that already points to a commit in our history.
+ for line in git_lines(
+ ["ls-remote", upstream, f"refs/tags/{firefox_tag_prefix}*"]
+ ):
+ objectname, name = line.split("\t", 1)
+ for index in range(search_index):
+ if search_commits[index] == objectname:
+ # Remove trailing "^{}" for commits pointed to by
+ # annotated tags.
+ remote_ref = Reference(re.sub(r"\^\{\}$", "", name), objectname)
+ # Only continue to search for references that are even
+ # closer to `search_from`.
+ search_index = index
+ break
+ if remote_ref is not None:
+ # Get a local copy of just this tag.
+ git_run(["fetch", "--no-tags", upstream, "tag", remote_ref.name])
+ return ref
+
+ raise TbDevException("Unable to find FIREFOX_ tag")
+
+
+def get_upstream_tracking_branch(search_from: str) -> str:
"""
- Search backwards from the 'search_from' commit to find the commit that comes
- from firefox.
+ :param search_from: The commit reference.
+ :returns: The upstream branch reference name.
"""
- return get_nearest_ref("tag", "FIREFOX_", search_from)
-
-
-def get_upstream_tracking_branch(search_from):
- return git_get(["rev-parse", "--abbrev-ref", f"{search_from}@{{upstream}}"])[0]
+ return git_get(["rev-parse", "--abbrev-ref", f"{search_from}@{{upstream}}"])
-def get_upstream_basis_commit(search_from):
+def get_upstream_basis_commit(search_from: str) -> str:
"""
Get the first common ancestor of search_from that is also in its upstream
branch.
+ :param search_from: The commit reference.
+ :returns: The upstream commit hash.
"""
upstream_branch = get_upstream_tracking_branch(search_from)
- commit = git_get(["merge-base", search_from, upstream_branch])[0]
+ commit = git_get(["merge-base", search_from, upstream_branch])
# Verify that the upstream commit shares the same firefox basis. Otherwise,
# this would indicate that the upstream is on an early or later FIREFOX
# base.
@@ -226,26 +341,82 @@ def get_upstream_basis_commit(search_from):
return commit
-def get_changed_files(from_commit, staged=False):
+class FileChange:
+ """Represents a git change to a commit."""
+
+ def __init__(self, status: str, path: str, new_path: str) -> None:
+ """
+ :param status: The file change status used within git diff. E.g. "M" for
+ modified, or "D" for deleted.
+ :param path: The source file path.
+ :param new_path: The file path after the change.
+ """
+ self.status = status
+ self.path = path
+ self.new_path = new_path
+
+
+RAW_DIFF_PATH_PATTERN = r"(?P<path>[^\0]*)\0"
+RAW_DIFF_LINE_REGEX = re.compile(
+ r":[0-7]+ [0-7]+ [0-9a-f]+ [0-9a-f]+ (?P<status>[ADMTUXRC])[0-9]*\0"
+ + RAW_DIFF_PATH_PATTERN
+)
+RAW_DIFF_PATH_REGEX = re.compile(RAW_DIFF_PATH_PATTERN)
+
+
+def parse_raw_diff_line(raw_output: str) -> tuple[FileChange, int]:
"""
- Get a list of filenames relative to the current working directory that have
+ Parse the --raw diff output from git.
+ :param raw_output: The raw output.
+ :returns: The change for this line, and the offset for the end of the raw
+ diff line.
+ """
+ match = RAW_DIFF_LINE_REGEX.match(raw_output)
+ if not match:
+ raise ValueError(f"Invalid raw output: {raw_output[:50]}...")
+ path = os.path.relpath(os.path.join(get_local_root(), match.group("path")))
+ status = match.group("status")
+ if status in ("R", "C"):
+ match = RAW_DIFF_PATH_REGEX.match(raw_output, pos=match.end())
+ if not match:
+ raise ValueError(f"Invalid raw output for rename: {raw_output[:50]}...")
+ new_path = os.path.relpath(os.path.join(get_local_root(), match.group("path")))
+ else:
+ new_path = path
+
+ return FileChange(status, path, new_path), match.end()
+
+
+def get_changed_files(
+ from_commit: None | str = None, staged: bool = False
+) -> Iterator[FileChange]:
+ """
+ Get a list of file changes relative to the current working directory that have
been changed since 'from_commit' (non-inclusive).
+ :param from_commit: The commit to compare against, otherwise use the git
+ diff default.
+ :param staged: Whether to limit the diff to staged changes.
+ :yield: The file changes.
"""
- args = ["diff"]
+ args = ["diff", "-z", "--raw"]
if staged:
args.append("--staged")
- args.append("--name-only")
- args.append(from_commit)
- return [
- os.path.relpath(os.path.join(get_local_root(), filename))
- for filename in git_get(args)
- ]
+ if from_commit:
+ args.append(from_commit)
+ raw_output = git_get(args, strip=False)
+ while raw_output:
+ file_change, end = parse_raw_diff_line(raw_output)
+ yield file_change
+ raw_output = raw_output[end:]
-def file_contains(filename, regex):
+def file_contains(filename: str, regex: re.Pattern[str]) -> bool:
"""
Return whether the file is a utf-8 text file containing the regular
expression given by 'regex'.
+ :param filename: The file path.
+ :param regex: The pattern to search for.
+ :returns: Whether the pattern was matched.
"""
with open(filename, encoding="utf-8") as file:
try:
@@ -258,9 +429,10 @@ def file_contains(filename, regex):
return False
-def get_gitlab_default():
+def get_gitlab_default() -> str:
"""
Get the name of the default branch on gitlab.
+ :returns: The branch name.
"""
repo_name = get_upstream_details().get("repo-name", None)
if repo_name is None:
@@ -283,12 +455,14 @@ def get_gitlab_default():
)
with urllib.request.urlopen(gitlab_request, timeout=20) as response:
- return json.load(response)["data"]["project"]["repository"]["rootRef"]
+ default = json.load(response)["data"]["project"]["repository"]["rootRef"]
+ assert isinstance(default, str)
+ return default
-def within_browser_root():
+def within_browser_root() -> bool:
"""
- Whether we are with the tor browser root.
+ :returns: Whether we are with the tor browser root.
"""
root = get_local_root()
if not root:
@@ -301,24 +475,24 @@ def within_browser_root():
# * -------------------- *
-def show_firefox_commit(_args):
+def show_firefox_commit(_args: argparse.Namespace) -> None:
"""
Print the tag name and commit for the last firefox commit below the current
HEAD.
"""
ref = get_firefox_ref("HEAD")
- print(ref.name)
+ print(ref.full_name)
print(ref.commit)
-def show_upstream_basis_commit(_args):
+def show_upstream_basis_commit(_args: argparse.Namespace) -> None:
"""
Print the last upstream commit for the current HEAD.
"""
print(get_upstream_basis_commit("HEAD"))
-def show_log(args):
+def show_log(args: argparse.Namespace) -> None:
"""
Show the git log between the current HEAD and the last firefox commit.
"""
@@ -326,7 +500,7 @@ def show_log(args):
git_run(["log", f"{commit}..HEAD", *args.gitargs], check=False)
-def show_files_containing(args):
+def show_files_containing(args: argparse.Namespace) -> None:
"""
List all the files that that have been modified for tor browser, that also
contain a regular expression.
@@ -336,33 +510,32 @@ def show_files_containing(args):
except re.error as err:
raise TbDevException(f"{args.regex} is not a valid python regex") from err
- file_list = get_changed_files(get_firefox_ref("HEAD").commit)
-
- for filename in file_list:
- if not os.path.isfile(filename):
+ for file_change in get_changed_files(get_firefox_ref("HEAD").commit):
+ path = file_change.new_path
+ if not os.path.isfile(path):
# deleted ofile
continue
- if file_contains(filename, regex):
- print(filename)
+ if file_contains(path, regex):
+ print(path)
-def show_changed_files(_args):
+def show_changed_files(_args: argparse.Namespace) -> None:
"""
List all the files that have been modified relative to upstream.
"""
- for filename in get_changed_files(get_upstream_basis_commit("HEAD")):
- print(filename)
+ for file_change in get_changed_files(get_upstream_basis_commit("HEAD")):
+ print(file_change.new_path)
-def lint_changed_files(args):
+def lint_changed_files(args: argparse.Namespace) -> None:
"""
Lint all the files that have been modified relative to upstream.
"""
os.chdir(get_local_root())
file_list = [
- f
+ f.new_path
for f in get_changed_files(get_upstream_basis_commit("HEAD"))
- if os.path.isfile(f) # Not deleted
+ if os.path.isfile(f.new_path) # Not deleted
]
# We add --warnings since clang only reports whitespace issues as warnings.
subprocess.run(
@@ -371,10 +544,18 @@ def lint_changed_files(args):
)
-def prompt_user(prompt, convert):
+# TODO: replace with "prompt_user[T](..., T]) -> T" after python 3.12 is the
+# minimum mach version.
+T = TypeVar("T")
+
+
+def prompt_user(prompt: str, convert: Callable[[str], T]) -> T:
"""
- Ask the user for some input until the given converter returns without
- throwing a ValueError.
+ Ask the user for some input.
+ :param prompt: The prompt to show the user.
+ :param convert: A method to convert the response into a type. Should
+ throw `ValueError` if the user should be re-prompted for a valid input.
+ :returns: The first valid user response.
"""
while True:
# Flush out stdin.
@@ -388,8 +569,12 @@ def prompt_user(prompt, convert):
pass
-def binary_reply_default_no(value):
- """Process a 'y' or 'n' reply, defaulting to 'n' if empty."""
+def binary_reply_default_no(value: str) -> bool:
+ """
+ Process a 'y' or 'n' reply, defaulting to 'n' if empty.
+ :param value: The user input.
+ :returns: Whether the answer is yes.
+ """
if value == "":
return False
if value.lower() == "y":
@@ -399,121 +584,737 @@ def binary_reply_default_no(value):
raise ValueError()
-def get_fixup_for_file(filename, firefox_commit):
- """Find the commit the given file should fix up."""
+class FixupTarget:
+ """Represents a commit that can be targeted by a fixup."""
+
+ def __init__(self, commit: str, short_ref: str, title: str) -> None:
+ """
+ :param commit: The commit hash for the commit.
+ :param short_ref: The shortened commit hash for display.
+ :param title: The first line of the commit message.
+ """
+ self.commit = commit
+ self.short_ref = short_ref
+ self.title = title
+ self.changes: list[FileChange] = []
+ self.fixups: list[FixupTarget] = []
+ self.target: None | FixupTarget = None
+
+ _FIXUP_REGEX = re.compile(r"^fixup! +")
+
+ def trim_fixup(self) -> tuple[str, int]:
+ """
+ Trim the "fixup!" prefixes.
+ :returns: The stripped commit title and the fixup depth (how many fixups
+ prefixes there were).
+ """
+ title = self.title
+ depth = 0
+ while True:
+ match = self.__class__._FIXUP_REGEX.match(title)
+ if not match:
+ return title, depth
+ title = title[match.end() :]
+ depth += 1
+
+ def touches_path(
+ self, path: str, filter_status: None | str = None, check_dir: bool = False
+ ) -> bool:
+ """
+ Whether this target, or one of its fixups or target, touches the given
+ path.
+ :param path: The path to check.
+ :param filter_status: Limit the detected changes to the given status(es).
+ :param check_dir: Whether we should treat `path` as a directory and check for
+ files within it.
+ :returns: Whether this target matches.
+ """
+ # NOTE: In the case of renames, we generally assume that renames occur
+ # in the fixup targets. E.g. "Commit 1" creates the file "file.txt", and
+ # "fixup! Commit 1" renames it to "new.txt". In this case, if the
+ # FixupTarget for "Commit 1" is passed in "file.txt" it will match. And
+ # if it is passed in "new.txt" it will also match via the self.fixups
+ # field, which will include the "fixup! Commit 1" rename.
+ # But the "fixup ! Commit 1" FixupTargets will only match with
+ # "file.txt" if they occurred before the rename fixup, and will only
+ # match with "new.txt" if they occur after the rename fixup. With the
+ # exception of the rename fixup itself, which will match both.
+ #
+ # In principle, we could identify a file across renames (have a mapping
+ # from each commit to what the file is called at that stage) and match
+ # using this file identifier. Similar to the "--follow" git diff
+ # argument. This would then cover cases where a rename occurs between
+ # the commit and its fixups, and allow fixups before the rename to also
+ # match. However, the former case is unexpected and the latter case
+ # would not be that useful.
+ if self._touches_path_basis(path, filter_status, check_dir):
+ return True
+ # Mark this as a valid target for the path if one of our fixups changes
+ # this path.
+ # NOTE: We use _touch_path_basis to prevent recursion. This means we
+ # will only check one layer up or down, but we only expect fixups of
+ # up to depth 1.
+ for fixup_target in self.fixups:
+ if fixup_target._touches_path_basis(path, filter_status, check_dir):
+ return True
+ # Mark this as a valid target if our target changes this path.
+ if self.target is not None and self.target._touches_path_basis(
+ path, filter_status, check_dir
+ ):
+ return True
+ return False
+
+ def _touches_path_basis(
+ self, path: str, filter_status: None | str, check_dir: bool
+ ) -> bool:
+ """
+ Whether this target touches the given path.
+ :param path: The path to check.
+ :param filter_status: Limit the detected changes to the given status.
+ :param check_dir: Whether we should treat `path` as a directory and check for
+ files within it.
+ :returns: Whether this target matches.
+ """
+ for file_change in self.changes:
+ if filter_status is not None and file_change.status not in filter_status:
+ continue
+ for test_path in (file_change.path, file_change.new_path):
+ if check_dir:
+ if os.path.commonpath((os.path.dirname(test_path), path)) == path:
+ # test_path's directory matches the path or is within it.
+ return True
+ elif test_path == path:
+ return True
+ return False
+
+
+def get_fixup_targets(
+ target_list: list[FixupTarget],
+ from_commit: str,
+ to_commit: str,
+ fixup_depth: int = 0,
+) -> None:
+ """
+ Find all the commits that can be targeted by a fixup between the given
+ commits.
+ :param target_list: The list to fill with targets. Appended in the order of
+ `from_commit` to `to_commit`.
+ :param from_commit: The commit to start from (non-inclusive).
+ :param to_commit: The commit to end on (inclusive).
+ :param fixup_depth: The maximum "depth" of fixups. I.e. how many "fixup!"
+ prefixes to allow.
+ """
+ raw_output = git_get(
+ [
+ "log",
+ "--pretty=format:%H,%h,%s",
+ "--reverse",
+ "--raw",
+ "-z",
+ f"{from_commit}..{to_commit}",
+ ],
+ strip=False,
+ )
+ pretty_regex = re.compile(
+ r"(?P<commit>[0-9a-f]+),(?P<short_ref>[0-9a-f]+),(?P<title>[^\n\0]*)\n"
+ )
+ excluded_regex_list = [
+ re.compile(r"^Bug [0-9]+.*r="), # Backported Mozilla bug.
+ re.compile(r"^dropme! "),
+ ]
+
+ while raw_output:
+ match = pretty_regex.match(raw_output)
+ if not match:
+ raise ValueError(f"Invalid pretty format: {raw_output[:100]}...")
+ fixup_target = FixupTarget(
+ match.group("commit"), match.group("short_ref"), match.group("title")
+ )
+ raw_output = raw_output[match.end() :]
+ while raw_output and raw_output[0] != "\0":
+ file_change, end = parse_raw_diff_line(raw_output)
+ fixup_target.changes.append(file_change)
+ raw_output = raw_output[end:]
+ if raw_output:
+ # Skip over the "\0".
+ raw_output = raw_output[1:]
+
+ for regex in excluded_regex_list:
+ if regex.match(fixup_target.title):
+ # Exclude from the list.
+ continue
+
+ trimmed_title, depth = fixup_target.trim_fixup()
+ if depth:
+ original_target = None
+ for target in target_list:
+ if target.title == trimmed_title:
+ original_target = target
+ break
+
+ if original_target:
+ original_target.fixups.append(fixup_target)
+ fixup_target.target = original_target
+ if depth > fixup_depth:
+ # Exclude from the list.
+ continue
+
+ target_list.append(fixup_target)
+
+
+class NewCommitBasis:
+ def __init__(self) -> None:
+ self.staged_paths: set[str] = set()
+ self.adding_paths: set[str] = set()
+
+ def add(self, paths: Iterable[str], staged: bool) -> None:
+ """
+ Add a path to include in this commit.
+ :param paths: The paths to add.
+ :param staged: Whether we are adding already staged changes.
+ """
+ if staged:
+ self.staged_paths.update(paths)
+ return
+
+ self.adding_paths.update(paths)
+
+
+class NewCommit(NewCommitBasis):
+ """Represents a new commit that we want to create."""
- def parse_log_line(line):
- [commit, short_ref, title] = line.split(",", 2)
- return {"commit": commit, "short-ref": short_ref, "title": title}
+ def __init__(self, alias: str) -> None:
+ """
+ :param alias: The alias name for the commit.
+ """
+ super().__init__()
+ self.alias = alias
- options = [
- parse_log_line(line)
- for line in git_get(
- [
- "log",
- "--pretty=format:%H,%h,%s",
- f"{firefox_commit}..HEAD",
- "--",
- filename,
- ]
+
+class NewFixup(NewCommitBasis):
+ """Represents a new fixup commit that we want to create."""
+
+ def __init__(self, target: FixupTarget) -> None:
+ """
+ :param target: The commit to target with the fixup.
+ """
+ super().__init__()
+ self.target = target
+
+
+def get_suggested_fixup_targets_for_change(
+ file_change: FileChange,
+ fixup_target_list: list[FixupTarget],
+ firefox_directories_lazy: Callable[[], set[str]],
+) -> Iterator[FixupTarget]:
+ """
+ Find the suggested fixup targets for the given file change.
+ :param file_change: The file change to get a suggestion for.
+ :param fixup_target_list: The list to choose from.
+ :param firefox_directories_lazy: Lazy method to return the firefox
+ directories.
+ :yield: The suggested fixup targets.
+ """
+
+ def filter_list(
+ path: str, filter_status: None | str = None, check_dir: bool = False
+ ) -> Iterator[FixupTarget]:
+ return (
+ t
+ for t in fixup_target_list
+ if t.touches_path(path, filter_status=filter_status, check_dir=check_dir)
)
+
+ if file_change.status == "D":
+ # Deleted.
+ # Find the commit that introduced this file or previously deleted it.
+ # I.e. added the file ("A"), renamed it ("R"), or deleted it ("D").
+ yield from filter_list(file_change.path, filter_status="ARD")
+ return
+
+ if file_change.status == "A":
+ # First check to see if this file name was actually touched before.
+ yielded_target = False
+ for target in filter_list(file_change.path):
+ yielded_target = True
+ yield target
+ if yielded_target:
+ return
+ # Else, find commits that introduced files in the same directory, or
+ # deleted in them, if they are not firefox directories.
+ dir_path = file_change.path
+ while True:
+ dir_path = os.path.dirname(dir_path)
+ if not dir_path or dir_path in firefox_directories_lazy():
+ return
+
+ yielded_target = False
+ for target in filter_list(dir_path, filter_status="ARD", check_dir=True):
+ yielded_target = True
+ yield target
+
+ if yielded_target:
+ return
+ # Else, search one directory higher.
+
+ if file_change.status == "R":
+ # Renamed.
+ # Find the commit that introduced the original name for this file.
+ yield from filter_list(file_change.path, filter_status="AR")
+ return
+
+ # Modified.
+ yield from filter_list(file_change.path)
+
+
+def ask_for_target(
+ file_change_list: list[FileChange],
+ new_commits_list: list[NewCommit | NewFixup],
+ suggested_fixup_target_list: list[FixupTarget],
+ full_fixup_target_list: list[FixupTarget],
+ staged: bool = False,
+) -> bool:
+ """
+ Ask the user to choose a target.
+ :param file_change_list: The file changes to ask for.
+ :param new_commits_list: The list of pending new commits, may be added to.
+ :param suggested_fixup_target_list: The list of suggested target fixups
+ to choose from.
+ :param staged: Whether this is for staged changes.
+ :returns: `True` if the operation should be aborted.
+ """
+
+ new_paths = [c.new_path for c in file_change_list]
+ all_paths = set(new_paths).union(c.path for c in file_change_list)
+ non_fixup_commits: list[NewCommit] = [
+ n for n in new_commits_list if isinstance(n, NewCommit)
]
- if not options:
- print(f"No commit found for {filename}")
- return None
- def valid_index(val):
+ shown_list: list[NewCommit | FixupTarget] = (
+ non_fixup_commits + suggested_fixup_target_list
+ )
+
+ can_skip = not staged
+ shown_full = False
+
+ index_offset = 2
+
+ def valid_response(val: str) -> tuple[str, None | NewCommit | FixupTarget]:
+ val = val.strip()
+
+ if val == "h":
+ return "help", None
+
+ if val == "a":
+ return "abort", None
+
if val == "d":
- return val
+ return "diff", None
+
+ if val == "f":
+ if shown_full:
+ # Already done once.
+ raise ValueError()
+ return "full-list", None
+ is_patch_full = val.startswith("P")
is_patch = val.startswith("p")
- if is_patch:
- val = val[1:]
+ if is_patch or is_patch_full:
+ index = int(val[1:], base=10) # Raises ValueError if not integer.
+ else:
+ index = int(val, base=10) # Raises ValueError if not integer.
+ if index == 0:
+ if not can_skip:
+ raise ValueError()
+ return "skip", None
+
+ if index == 1:
+ return "new", None
- # May raise a ValueError.
- as_index = int(val)
- if as_index < 0 or as_index > len(options):
+ index -= index_offset
+
+ if index < 0 or index >= len(shown_list):
raise ValueError()
- if as_index == 0:
- if is_patch:
+ selected = shown_list[index]
+
+ if is_patch_full:
+ return "patch-full", selected
+ if is_patch:
+ return "patch", selected
+ return "target", selected
+
+ def alias_response(val: str) -> str:
+ # Choose a default alias name if none is given.
+ val = val.strip() or f"New commit {len(non_fixup_commits)}"
+ for new_commit in non_fixup_commits:
+ if new_commit.alias == val:
+ # Already in use.
raise ValueError()
- return None
+ return val
+
+ def print_index_option(index: int, description: str) -> None:
+ print(f" \x1b[1m{index}\x1b[0m: {description}")
- return (is_patch, options[as_index - 1]["commit"])
+ def in_pink(text: str) -> str:
+ return f"\x1b[1;38;5;212m{text}\x1b[0m"
+ prefix_str = "For " + (in_pink("staged") if staged else "unstaged") + " changes to"
+ if len(new_paths) == 1:
+ print(f"{prefix_str} {in_pink(new_paths[0])}:")
+ else:
+ print(f"{prefix_str}:")
+ for path in new_paths:
+ print(f" {in_pink(path)}")
+ print("")
+
+ show_help = True
+ reshow_list = True
while True:
- print(f"For {filename}:\n")
- print(" \x1b[1m0\x1b[0m: None")
- for index, opt in enumerate(options):
- print(
- f" \x1b[1m{index + 1}\x1b[0m: "
- + f"\x1b[1;38;5;212m{opt['short-ref']}\x1b[0m "
- + opt["title"]
- )
+ if reshow_list:
+ if can_skip:
+ print_index_option(0, "Skip")
+ print_index_option(1, "New commit")
+ for index, target in enumerate(shown_list, start=index_offset):
+ if isinstance(target, NewCommit):
+ print_index_option(index, f"Add to new commit: {target.alias}")
+ else:
+ print_index_option(
+ index, f"Fixup: {in_pink(target.short_ref)} {target.title}"
+ )
+ reshow_list = False
print("")
- response = prompt_user(
- "Choose an <index> to fixup, or '0' to skip this file, "
- "or 'd' to view the pending diff, "
- "or 'p<index>' to view the patch for the index: ",
- valid_index,
+
+ response, selected = prompt_user(
+ (
+ "Choose an <index> to target. Type 'h' for additional options: "
+ if show_help
+ else "Choose an <index> to target or an option: "
+ ),
+ valid_response,
)
- if response is None:
- # Skip this file.
- return None
- if response == "d":
- git_run(["diff", "--", filename])
+ if response == "help":
+ print("Options:")
+ for option, desc in (
+ ("h", "show the available options."),
+ ("a", "abort this commit operation and all pending commits."),
+ (
+ ("", "")
+ if shown_full
+ else (
+ "f",
+ "show the full list of fixup targets, rather than just the suggested ones.",
+ )
+ ),
+ ("d", "view the diff for the pending file changes."),
+ (
+ "P<index>",
+ "view the patch for the index (including its relevant fixups).",
+ ),
+ (
+ "p<index>",
+ "view the patch for the index (including its relevant fixups), "
+ "limited to the current files.",
+ ),
+ ):
+ if not option:
+ # Skip this option.
+ continue
+ print(f" \x1b[1m{option[0]}\x1b[0m{option[1:].ljust(7)}: {desc}")
+ # Do not show the help option again.
+ show_help = False
+ continue
+
+ if response == "abort":
+ return True
+
+ if response == "skip":
+ return False
+
+ if response == "new":
+ new_alias = prompt_user(
+ "Enter an optional temporary alias for this new commit: ",
+ alias_response,
+ )
+ new_commit = NewCommit(new_alias)
+ new_commit.add(all_paths, staged)
+ new_commits_list.append(new_commit)
+ return False
+
+ if response == "target":
+ assert selected is not None
+
+ if isinstance(selected, NewCommit):
+ # Adding to a new commit.
+ selected.add(all_paths, staged)
+ return False
+
+ for new_fixup in new_commits_list:
+ if not isinstance(new_fixup, NewFixup):
+ continue
+ if new_fixup.target == selected:
+ # We already have a pending fixup commit that targets this
+ # selected target. Add this path to the same commit.
+ new_fixup.add(all_paths, staged)
+ return False
+
+ new_fixup = NewFixup(selected)
+ new_fixup.add(all_paths, staged)
+ new_commits_list.append(new_fixup)
+ return False
+
+ if response == "full-list":
+ shown_list = non_fixup_commits + full_fixup_target_list
+ shown_full = True
+ reshow_list = True
continue
- view_patch, commit = response
- if view_patch:
- git_run(["log", "-p", "-1", commit, "--", filename])
+ if response == "diff":
+ git_args = ["diff", "--color"]
+ if staged:
+ git_args.append("--staged")
+ git_args.extend(git_path_args(all_paths))
+ git_run_pager(git_args)
continue
- return commit
+ if response in ("patch", "patch-full"):
+ assert selected is not None
+
+ filter_paths = response == "patch"
+
+ if isinstance(selected, NewCommit):
+ git_sequence = [
+ ["diff", "--color", "--staged", *git_path_args((path,))]
+ for path in selected.staged_paths
+ if not filter_paths or path in all_paths
+ ]
+ git_sequence.extend(
+ ["diff", "--color", *git_path_args((path,))]
+ for path in selected.adding_paths
+ if not filter_paths or path in all_paths
+ )
+
+ # Show what the expected patch will be for the new commit.
+ git_run_pager(
+ arg_sequence=git_sequence, pager_prefix=f"{selected.alias}\n\n"
+ )
+ else:
+ # Show the log entry for the FixupTarget and each of its fixups.
+ # Order with the commmit closest to HEAD first. We expect
+ # selected.fixups to match this order.
+ git_sequence = []
+ # If `filter_paths` is set, we want to limit the log to the
+ # paths, and try to track any renames in the commit history.
+ prev_log_paths: None | set[str] = None
+ # For the first commit in the sequence, we use the old path
+ # names (rather than `c.new_path`) since we expect the commit
+ # which is closest to us to use the older names.
+ log_paths: None | set[str] = (
+ {c.path for c in file_change_list} if filter_paths else None
+ )
+ for target in (*selected.fixups, selected):
+ git_args = [
+ "log",
+ "--color",
+ "-p",
+ f"{target.commit}~1..{target.commit}",
+ ]
+ if filter_paths:
+ assert log_paths is not None
+ # Track the renamed paths.
+ prev_log_paths = log_paths.copy()
+ for file_change in target.changes:
+ if (
+ file_change.status == "R"
+ and file_change.new_path in log_paths
+ ):
+ # file was renamed in this change.
+ # Update log_paths to the new name.
+ # NOTE: This should have a similar effect to the
+ # --follow option for git log for a single file
+ # NOTE: File renames will not be properly
+ # tracked if a rename occurs outside of
+ # `selected.changes` or
+ # `selected.fixups[].changes`, but this is
+ # unexpected.
+ log_paths.remove(file_change.new_path)
+ log_paths.add(file_change.path)
+
+ # NOTE: This log entry may be empty if none of the paths
+ # match.
+ # NOTE: We include both log_paths and prev_log_paths to
+ # show renames in the diff output.
+ git_args.extend(git_path_args(log_paths | prev_log_paths))
+ git_sequence.append(git_args)
+ # Combine all the logs into one.
+ git_run_pager(arg_sequence=git_sequence)
+ continue
+
+ raise ValueError(f"Unexpected response: {response}")
-def auto_fixup(_args):
+def auto_commit(_args: argparse.Namespace) -> None:
"""
- Automatically find and fix up commits using the current unstaged changes.
+ Automatically find and fix up commits for any pending changes.
"""
+ # Want git log and add to be run from the root.
+ os.chdir(get_local_root())
# Only want to search as far back as the firefox commit.
firefox_commit = get_firefox_ref("HEAD").commit
- staged_files = get_changed_files("HEAD", staged=True)
- if staged_files:
- raise TbDevException(f"Have already staged files: {staged_files}")
+ staged_changes = [f for f in get_changed_files(staged=True)]
+ if staged_changes:
+ print("Existing staged changes for:")
+ for file_change in staged_changes:
+ print(f" {file_change.new_path}")
+ if not prompt_user(
+ "Include staged changes? (y/\x1b[4mn\x1b[0m)", binary_reply_default_no
+ ):
+ raise TbDevException("Cannot continue with pending staged changes")
+ print("")
- fixups = {}
- for filename in get_changed_files("HEAD"):
- commit = get_fixup_for_file(filename, firefox_commit)
- if commit is None:
+ full_target_list: list[FixupTarget] = []
+ # Determine if HEAD points to a branch or not and has an upstream commit.
+ # We choose check=False since the exit status is non-zero when we are in a
+ # detached state.
+ head_symbolic_ref = git_get(["symbolic-ref", "-q", "HEAD"], check=False)
+ if not head_symbolic_ref or not bool(
+ git_get(["for-each-ref", "--format=%(upstream)", head_symbolic_ref])
+ ):
+ # Unexpected, but not fatal.
+ print("HEAD has no upstream tracking!")
+ # Just include all commits since firefox_commit with no fixup depth
+ get_fixup_targets(full_target_list, firefox_commit, "HEAD", fixup_depth=0)
+ else:
+ upstream_commit = get_upstream_basis_commit("HEAD")
+ # Only include "fixup!" commits that are between here and the upstream
+ # tracking commit.
+ get_fixup_targets(
+ full_target_list, firefox_commit, upstream_commit, fixup_depth=0
+ )
+ get_fixup_targets(full_target_list, upstream_commit, "HEAD", fixup_depth=1)
+
+ # full_target_list is ordered with the earlier commits first. Reverse this.
+ full_target_list.reverse()
+ # Also reverse the fixups order to follow the same order.
+ for target in full_target_list:
+ target.fixups.reverse()
+
+ # Lazy load the list of firefox directories since they are unlikely to be
+ # needed.
+ @functools.cache
+ def firefox_directories_lazy() -> set[str]:
+ return {
+ dir_name
+ for dir_name in git_get(
+ [
+ "ls-tree",
+ "-r",
+ "-d",
+ "--name-only",
+ "--full-tree",
+ "-z",
+ firefox_commit,
+ ],
+ strip=False,
+ ).split("\0")
+ if dir_name
+ }
+
+ # Check untracked files to be added.
+ for path in git_get(
+ ["ls-files", "--other", "--exclude-standard", "-z"], strip=False
+ ).split("\0"):
+ if not path:
continue
- if commit not in fixups:
- fixups[commit] = [filename]
- else:
- fixups[commit].append(filename)
+ if prompt_user(
+ f"Start tracking file `{path}`? (y/\x1b[4mn\x1b[0m)",
+ binary_reply_default_no,
+ ):
+ # Include in the git diff output, but do not stage.
+ git_run(["add", "--intent-to-add", path])
print("")
- for commit, files in fixups.items():
- print("")
- git_run(["add", *files])
- git_run(["commit", f"--fixup={commit}"])
+ aborted = False
+ new_commits_list: list[NewCommit | NewFixup] = []
+ # First go through staged changes.
+ if staged_changes:
+ common_fixup_targets = None
+ for change in staged_changes:
+ target_iter = get_suggested_fixup_targets_for_change(
+ change, full_target_list, firefox_directories_lazy
+ )
+ if common_fixup_targets is None:
+ common_fixup_targets = set(target_iter)
+ else:
+ common_fixup_targets.intersection_update(target_iter)
+
+ assert common_fixup_targets is not None
+
+ aborted = ask_for_target(
+ staged_changes,
+ new_commits_list,
+ # Sort in the same order as full_target_list.
+ [target for target in full_target_list if target in common_fixup_targets],
+ full_target_list,
+ staged=True,
+ )
print("")
- if prompt_user(
- "Edit fixup commit message? (y/\x1b[4mn\x1b[0m)", binary_reply_default_no
- ):
+ if not aborted:
+ for file_change in get_changed_files():
+ target_list = list(
+ get_suggested_fixup_targets_for_change(
+ file_change, full_target_list, firefox_directories_lazy
+ )
+ )
+ aborted = ask_for_target(
+ [file_change],
+ new_commits_list,
+ target_list,
+ full_target_list,
+ staged=False,
+ )
+ print("")
+ if aborted:
+ break
+
+ if aborted:
+ return
+
+ # NOTE: Only the first commit can include staged changes.
+ # This should already be the case, but we want to double check.
+ for commit_index in range(1, len(new_commits_list)):
+ if new_commits_list[commit_index].staged_paths:
+ raise ValueError(f"Staged changes for commit {commit_index}")
+
+ for new_commit in new_commits_list:
+ print("")
+ if new_commit.adding_paths:
+ git_run(["add", *git_path_args(new_commit.adding_paths)])
+ if isinstance(new_commit, NewFixup):
+ git_run(["commit", f"--fixup={new_commit.target.commit}"])
+ print("")
+ is_double_fixup = bool(new_commit.target.target)
+ if not is_double_fixup and prompt_user(
+ "Edit fixup commit message? (y/\x1b[4mn\x1b[0m)",
+ binary_reply_default_no,
+ ):
+ git_run(["commit", "--amend"])
+ print("")
+ else:
+ git_run(["commit", "-m", new_commit.alias])
git_run(["commit", "--amend"])
+ print("")
-def clean_fixups(_args):
+def clean_fixups(_args: argparse.Namespace) -> None:
"""
Perform an interactive rebase that automatically applies fixups, similar to
--autosquash but also works on fixups of fixups.
"""
- user_editor = git_get(["var", "GIT_SEQUENCE_EDITOR"])[0]
+ user_editor = git_get(["var", "GIT_SEQUENCE_EDITOR"])
sub_editor = os.path.join(
os.path.dirname(os.path.realpath(__file__)), FIXUP_PREPROCESSOR_EDITOR
)
@@ -525,7 +1326,7 @@ def clean_fixups(_args):
)
-def show_default(_args):
+def show_default(_args: argparse.Namespace) -> None:
"""
Print the default branch name from gitlab.
"""
@@ -536,7 +1337,7 @@ def show_default(_args):
print(f"{upstream}/{default_branch}")
-def branch_from_default(args):
+def branch_from_default(args: argparse.Namespace) -> None:
"""
Fetch the default gitlab branch from upstream and create a new local branch.
"""
@@ -557,7 +1358,7 @@ def branch_from_default(args):
)
-def move_to_default(args):
+def move_to_default(args: argparse.Namespace) -> None:
"""
Fetch the default gitlab branch from upstream and move the specified
branch's commits on top. A new branch will be created tracking the default
@@ -569,7 +1370,7 @@ def move_to_default(args):
if branch_name is None:
# Use current branch as default.
try:
- branch_name = git_get(["branch", "--show-current"])[0]
+ branch_name = git_get(["branch", "--show-current"])
except IndexError:
raise TbDevException("No current branch")
@@ -608,7 +1409,7 @@ def move_to_default(args):
git_run(["cherry-pick", f"{current_basis}..{old_branch_name}"], check=False)
-def show_range_diff(args):
+def show_range_diff(args: argparse.Namespace) -> None:
"""
Show the range diff between two branches, from their firefox bases.
"""
@@ -624,21 +1425,21 @@ def show_range_diff(args):
)
-def show_diff_diff(args):
+def show_diff_diff(args: argparse.Namespace) -> None:
"""
Show the diff between the diffs of two branches, relative to their firefox
bases.
"""
- config_res = git_get(["config", "--get", "diff.tool"])
- if not config_res:
+ try:
+ diff_tool = next(git_lines(["config", "--get", "diff.tool"]))
+ except StopIteration:
raise TbDevException("No diff.tool configured for git")
- diff_tool = config_res[0]
# Filter out parts of the diff we expect to be different.
index_regex = re.compile(r"index [0-9a-f]{12}\.\.[0-9a-f]{12}")
lines_regex = re.compile(r"@@ -[0-9]+,[0-9]+ \+[0-9]+,[0-9]+ @@(?P<rest>.*)")
- def save_diff(branch):
+ def save_diff(branch: str) -> str:
firefox_commit = get_firefox_ref(branch).commit
file_desc, file_name = tempfile.mkstemp(
text=True, prefix=f'{branch.split("/")[-1]}-'
@@ -653,6 +1454,7 @@ def show_diff_diff(args):
)
with os.fdopen(file_desc, "w") as file:
+ assert diff_process.stdout is not None
for line in diff_process.stdout:
if index_regex.match(line):
# Fake data that will match.
@@ -665,7 +1467,7 @@ def show_diff_diff(args):
continue
file.write(line)
- status = diff_process.poll()
+ status = diff_process.wait()
if status != 0:
raise TbDevException(f"git diff exited with status {status}")
@@ -681,7 +1483,7 @@ def show_diff_diff(args):
# * -------------------- *
-def branch_complete(prefix, parsed_args, **kwargs):
+def branch_complete(prefix: str, **_kwargs: Any) -> list[str]:
"""
Complete the argument with a branch name.
"""
@@ -689,7 +1491,7 @@ def branch_complete(prefix, parsed_args, **kwargs):
return []
try:
branches = [ref.name for ref in get_refs("head", "")]
- branches.extend([ref.name for ref in get_refs("remote", "")])
+ branches.extend(ref.name for ref in get_refs("remote", ""))
branches.append("HEAD")
except Exception:
return []
@@ -699,7 +1501,20 @@ def branch_complete(prefix, parsed_args, **kwargs):
parser = argparse.ArgumentParser()
subparsers = parser.add_subparsers(required=True)
-for name, details in {
+
+class ArgConfig(TypedDict):
+ help: str
+ metavar: NotRequired[str]
+ nargs: NotRequired[str]
+ completer: NotRequired[Callable[[str], list[str]]]
+
+
+class CommandConfig(TypedDict):
+ func: Callable[[argparse.Namespace], None]
+ args: NotRequired[dict[str, ArgConfig]]
+
+
+all_commands: dict[str, CommandConfig] = {
"show-upstream-basis-commit": {
"func": show_upstream_basis_commit,
},
@@ -716,8 +1531,8 @@ for name, details in {
},
},
},
- "auto-fixup": {
- "func": auto_fixup,
+ "auto-commit": {
+ "func": auto_commit,
},
"clean-fixups": {
"func": clean_fixups,
@@ -794,20 +1609,25 @@ for name, details in {
"regex": {"help": "the regex that the files must contain"},
},
},
-}.items():
- help_message = re.sub(r"\s+", " ", details["func"].__doc__).strip()
+}
+
+for name, command_config in all_commands.items():
+ help_message = command_config["func"].__doc__
+ assert isinstance(help_message, str)
+ help_message = re.sub(r"\s+", " ", help_message).strip()
sub = subparsers.add_parser(name, help=help_message)
- sub.set_defaults(func=details["func"])
- for arg, keywords in details.get("args", {}).items():
+ sub.set_defaults(func=command_config["func"])
+ for arg, keywords in command_config.get("args", {}).items():
completer = None
if "completer" in keywords:
completer = keywords["completer"]
del keywords["completer"]
sub_arg = sub.add_argument(arg, **keywords)
- if completer is not None:
- sub_arg.completer = completer
+ if completer is not None and argcomplete is not None:
+ sub_arg.completer = completer # type: ignore
-argcomplete.autocomplete(parser)
+if argcomplete is not None:
+ argcomplete.autocomplete(parser)
try:
if not within_browser_root():
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/9596a5…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/9596a5…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/mullvad-browser][mullvad-browser-146.0a1-16.0-2] MB 80: Enable Mullvad Browser as a default browser
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
11 Dec '25
Pier Angelo Vendrame pushed to branch mullvad-browser-146.0a1-16.0-2 at The Tor Project / Applications / Mullvad Browser
Commits:
0801264b by Pier Angelo Vendrame at 2025-12-11T12:21:20+01:00
MB 80: Enable Mullvad Browser as a default browser
- - - - -
10 changed files:
- browser/components/shell/ShellService.sys.mjs
- browser/components/shell/WindowsDefaultBrowser.cpp
- browser/components/shell/nsWindowsShellService.cpp
- + other-licenses/nsis/Contrib/ApplicationID/Makefile
- other-licenses/nsis/Contrib/ApplicationID/Set.cpp
- + other-licenses/nsis/Contrib/CityHash/Makefile
- toolkit/mozapps/defaultagent/EventLog.h
- toolkit/mozapps/defaultagent/SetDefaultBrowser.cpp
- widget/windows/WinTaskbar.cpp
- widget/windows/moz.build
Changes:
=====================================
browser/components/shell/ShellService.sys.mjs
=====================================
@@ -286,7 +286,7 @@ let ShellServiceInternal = {
) {
if (this._shouldSetDefaultPDFHandler()) {
lazy.log.info("Setting Firefox as default PDF handler");
- extraFileExtensions.push(".pdf", "FirefoxPDF");
+ extraFileExtensions.push(".pdf", "MullvadBrowserPDF");
} else {
lazy.log.info("Not setting Firefox as default PDF handler");
}
@@ -324,7 +324,7 @@ let ShellServiceInternal = {
try {
this.defaultAgent.setDefaultExtensionHandlersUserChoice(aumi, [
".pdf",
- "FirefoxPDF",
+ "MullvadBrowserPDF",
]);
} catch (err) {
telemetryResult = "ErrOther";
=====================================
browser/components/shell/WindowsDefaultBrowser.cpp
=====================================
@@ -24,7 +24,7 @@
#include <wchar.h>
#include <windows.h>
-#define APP_REG_NAME_BASE L"Firefox-"
+#define APP_REG_NAME_BASE L"MullvadBrowser-"
static bool IsWindowsLogonConnected() {
WCHAR userName[UNLEN + 1];
=====================================
browser/components/shell/nsWindowsShellService.cpp
=====================================
@@ -361,10 +361,12 @@ nsWindowsShellService::CheckAllProgIDsExist(bool* aResult) {
*aResult = result;
} else {
- *aResult =
- CheckProgIDExists(FormatProgID(L"FirefoxURL", aumid.get()).get()) &&
- CheckProgIDExists(FormatProgID(L"FirefoxHTML", aumid.get()).get()) &&
- CheckProgIDExists(FormatProgID(L"FirefoxPDF", aumid.get()).get());
+ *aResult = CheckProgIDExists(
+ FormatProgID(L"MullvadBrowserURL", aumid.get()).get()) &&
+ CheckProgIDExists(
+ FormatProgID(L"MullvadBrowserHTML", aumid.get()).get()) &&
+ CheckProgIDExists(
+ FormatProgID(L"MullvadBrowserPDF", aumid.get()).get());
}
return NS_OK;
=====================================
other-licenses/nsis/Contrib/ApplicationID/Makefile
=====================================
@@ -0,0 +1,14 @@
+CXXFLAGS=-Icityhash -DWIN32 -DNDEBUG -D_WINDOWS -D_USRDLL -DCITYHASH_EXPORTS -DUNICODE
+SRCS=Set.cpp
+OBJS=$(subst .cpp,.o,$(SRCS))
+LDFLAGS=-lole32 -lshlwapi -shared -Wl,--no-insert-timestamp
+PLUGIN=ApplicationID.dll
+
+all: $(PLUGIN)
+
+$(PLUGIN): $(OBJS)
+ $(CXX) $(OBJS) $(LDFLAGS) -o $@
+ llvm-strip $@
+
+clean:
+ $(RM) $(OBJS) $(PLUGIN)
=====================================
other-licenses/nsis/Contrib/ApplicationID/Set.cpp
=====================================
@@ -35,7 +35,8 @@ unsigned int g_stringsize;
TCHAR *g_variables;
// Indicates that an application supports dual desktop and immersive modes. In Windows 8, this property is only applicable for web browsers.
-DEFINE_PROPERTYKEY(PKEY_AppUserModel_IsDualMode, 0x9F4C2855, 0x9F79, 0x4B39, 0xA8, 0xD0, 0xE1, 0xD4, 0x2D, 0xE1, 0xD5, 0xF3, 11);
+// mingw already defines this in propkey.h.
+// DEFINE_PROPERTYKEY(PKEY_AppUserModel_IsDualMode, 0x9F4C2855, 0x9F79, 0x4B39, 0xA8, 0xD0, 0xE1, 0xD4, 0x2D, 0xE1, 0xD5, 0xF3, 11);
int popstring(TCHAR *str, int len);
void pushstring(const TCHAR *str, int len);
=====================================
other-licenses/nsis/Contrib/CityHash/Makefile
=====================================
@@ -0,0 +1,14 @@
+CXXFLAGS=-Icityhash -DWIN32 -DNDEBUG -D_WINDOWS -D_USRDLL -DCITYHASH_EXPORTS -DUNICODE
+SRCS=CityHash.cpp cityhash/city.cpp
+OBJS=$(subst .cpp,.o,$(SRCS))
+LDFLAGS=-shared -Wl,--no-insert-timestamp
+PLUGIN=CityHash.dll
+
+all: $(PLUGIN)
+
+$(PLUGIN): $(OBJS)
+ $(CXX) $(OBJS) $(LDFLAGS) -o $@
+ llvm-strip $@
+
+clean:
+ $(RM) $(OBJS) $(PLUGIN)
=====================================
toolkit/mozapps/defaultagent/EventLog.h
=====================================
@@ -7,7 +7,7 @@
#ifndef __DEFAULT_BROWSER_AGENT_EVENT_LOG_H__
#define __DEFAULT_BROWSER_AGENT_EVENT_LOG_H__
-#include "mozilla/Types.h"
+#include <cwchar>
MOZ_BEGIN_EXTERN_C
@@ -15,10 +15,21 @@ extern MOZ_EXPORT const wchar_t* gWinEventLogSourceName;
MOZ_END_EXTERN_C
-#include "mozilla/WindowsEventLog.h"
-
-#define LOG_ERROR(hr) MOZ_WIN_EVENT_LOG_ERROR(gWinEventLogSourceName, hr)
-#define LOG_ERROR_MESSAGE(format, ...) \
- MOZ_WIN_EVENT_LOG_ERROR_MESSAGE(gWinEventLogSourceName, format, __VA_ARGS__)
+#ifdef LOG_ERRORS_FILE
+extern FILE* gLogFile;
+# define LOG_ERROR(hr) \
+ if (gLogFile) { \
+ fprintf(gLogFile, "Error in %s:%d: 0x%X\r\n", __FILE__, __LINE__, \
+ (unsigned int)hr); \
+ }
+# define LOG_ERROR_MESSAGE(format, ...) \
+ if (gLogFile) { \
+ fwprintf(gLogFile, format __VA_OPT__(, ) __VA_ARGS__); \
+ fputs("\r\n", gLogFile); \
+ }
+#else
+# define LOG_ERROR(hr)
+# define LOG_ERROR_MESSAGE(format, ...)
+#endif
#endif // __DEFAULT_BROWSER_AGENT_EVENT_LOG_H__
=====================================
toolkit/mozapps/defaultagent/SetDefaultBrowser.cpp
=====================================
@@ -444,9 +444,10 @@ nsresult SetDefaultBrowserUserChoice(
return NS_ERROR_FAILURE;
}
- nsTArray<nsString> browserDefaults = {
- u"https"_ns, u"FirefoxURL"_ns, u"http"_ns, u"FirefoxURL"_ns,
- u".html"_ns, u"FirefoxHTML"_ns, u".htm"_ns, u"FirefoxHTML"_ns};
+ nsTArray<nsString> browserDefaults = {u"https"_ns, u"MullvadBrowserURL"_ns,
+ u"http"_ns, u"MullvadBrowserURL"_ns,
+ u".html"_ns, u"MullvadBrowserHTML"_ns,
+ u".htm"_ns, u"MullvadBrowserHTML"_ns};
browserDefaults.AppendElements(aExtraFileExtensions);
=====================================
widget/windows/WinTaskbar.cpp
=====================================
@@ -249,7 +249,7 @@ bool WinTaskbar::GenerateAppUserModelID(nsAString& aAppUserModelId,
nsCString appName;
if (appInfo && NS_SUCCEEDED(appInfo->GetName(appName))) {
nsAutoString regKey;
- regKey.AssignLiteral("Software\\Mozilla\\");
+ regKey.AssignLiteral("Software\\" MOZ_APP_VENDOR "\\");
AppendASCIItoUTF16(appName, regKey);
regKey.AppendLiteral("\\TaskBarIDs");
=====================================
widget/windows/moz.build
=====================================
@@ -209,6 +209,7 @@ DEFINES["MOZ_UNICODE"] = True
DEFINES["MOZ_APP_NAME"] = '"%s"' % CONFIG["MOZ_APP_NAME"]
# Turn `firefox` into `Firefox`.
DEFINES["MOZ_TOAST_APP_NAME"] = '"%s"' % CONFIG["MOZ_APP_NAME"].title()
+DEFINES["MOZ_APP_VENDOR"] = '"%s"' % CONFIG["MOZ_APP_VENDOR"]
USE_LIBS += [
"jsoncpp",
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/commit/080…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/commit/080…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/tor-browser-build][main] 2 commits: Bug 41627: Build OpenSSL in the Python package.
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
11 Dec '25
Pier Angelo Vendrame pushed to branch main at The Tor Project / Applications / tor-browser-build
Commits:
b0b67b81 by Pier Angelo Vendrame at 2025-12-10T21:38:25+01:00
Bug 41627: Build OpenSSL in the Python package.
Build OpenSSL with Python, so that the ssl module can be imported.
We do not use the openssl project to avoid re-building a consistent
part of the Linux toolchain every time we bump it.
Also, bump the Python version since we are at it (we downgraded to be
able to build it with OpenSSL 1.1.x, now we don't have that requirement
anymore), and bump ninja, as the old version didn't build with our new
version of Python.
- - - - -
75825497 by Pier Angelo Vendrame at 2025-12-10T21:39:51+01:00
Bug 41662: Add python-zstandard to desktop containers.
They are needed when we build artifacts.
- - - - -
12 changed files:
- projects/common/list_toolchain_updates-common-firefox-geckoview
- projects/firefox/build
- projects/firefox/config
- projects/ninja/build
- projects/ninja/config
- − projects/ninja/python3.patch
- + projects/python-zstandard/README.md
- + projects/python-zstandard/build
- + projects/python-zstandard/config
- projects/python/README.md
- projects/python/build
- projects/python/config
Changes:
=====================================
projects/common/list_toolchain_updates-common-firefox-geckoview
=====================================
@@ -159,7 +159,7 @@ if (m/^\\s*MINIMUM_MINOR_VERSION = ([0-9]+)/) {
}
EOF
needed=3.$(cat python/mozboot/bin/bootstrap.py | perl -ne "$p")
-current="3.9" # 3.11.x on Debian bookworm, 3.9.20 on our python project
+current="3.9" # 3.11.x on Debian bookworm, 3.13.11 on our python project
check_update_needed python "$needed" "$current"
=====================================
projects/firefox/build
=====================================
@@ -34,6 +34,8 @@ export PATH="/var/tmp/dist/rust/bin:/var/tmp/dist/cbindgen:/var/tmp/dist/node/bi
tar -C /var/tmp/dist -xf [% c('input_files_by_name/clang') %]
tar -C /var/tmp/dist -xf [% c('input_files_by_name/python') %]
export PATH="/var/tmp/dist/python/bin:$PATH"
+ # For OpenSSL, see Python's README.md.
+ export LD_LIBRARY_PATH=/var/tmp/dist/python/lib:$LD_LIBRARY_PATH
[% IF ! c("var/linux-cross") -%]
tar -C /var/tmp/dist -xf $rootdir/[% c('input_files_by_name/binutils') %]
export PATH="/var/tmp/dist/binutils/bin:$PATH"
@@ -48,6 +50,9 @@ export PATH="/var/tmp/dist/rust/bin:/var/tmp/dist/cbindgen:/var/tmp/dist/node/bi
# down properly in that case. Thus, we set it here in the build script.
export PKG_CONFIG_PATH="${PKG_CONFIG_PATH}:/usr/lib/[% c("var/crosstarget") %]/pkgconfig"
[% END -%]
+ [% IF c("var/dev_artifacts") -%]
+ python3 -m pip install $rootdir/[% c('input_files_by_name/python-zstandard') %]/*.whl
+ [% END -%]
[% END -%]
[% IF c("var/macos") && c("var/dev_artifacts") %]
=====================================
projects/firefox/config
=====================================
@@ -154,6 +154,7 @@ targets:
arch_deps:
- python3
- python3-distutils
+ - python3-zstandard
- rsync
windows:
@@ -161,6 +162,7 @@ targets:
arch_deps:
- python3
- python3-distutils
+ - python3-zstandard
- wine
input_files:
@@ -228,6 +230,9 @@ input_files:
enable: '[% c("var/mullvad-browser") && c("var/has_l10n") %]'
- filename: marsigner.der
enable: '[% c("var/override_updater_url") %]'
+ - project: python-zstandard
+ enable: '[% c("var/linux") && c("var/dev_artifacts") %]'
+ name: python-zstandard
- filename: dmg-root
enable: '[% c("var/macos") && c("var/dev_artifacts") %]'
- project: hfsplus-tools
=====================================
projects/ninja/build
=====================================
@@ -12,8 +12,6 @@ mkdir -p /var/tmp/build
tar -C /var/tmp/build -xf [% project %]-[% c('version') %].tar.[% c('compress_tar') %]
cd /var/tmp/build/[% project %]-[% c('version') %]
-patch -p1 < "$rootdir/python3.patch"
-
./configure.py --bootstrap
mkdir -p $distdir
=====================================
projects/ninja/config
=====================================
@@ -1,7 +1,7 @@
# vim: filetype=yaml sw=2
-version: '[% c("abbrev") %]'
+version: '1.13.2'
git_url: https://github.com/ninja-build/ninja.git
-git_hash: a524bf3f6bacd1b4ad85d719eed2737d8562f27a #v1.11.1
+git_hash: 3441b633c2fe2c494e958780ba0f4227b1327634 # v1.13.2
filename: '[% project %]-[% c("version") %]-[% c("var/build_id") %].tar.[% c("compress_tar") %]'
container:
use_container: 1
@@ -16,4 +16,3 @@ input_files:
- name: python
project: python
enable: '[% c("var/linux") %]'
- - filename: python3.patch
=====================================
projects/ninja/python3.patch deleted
=====================================
@@ -1,95 +0,0 @@
-From 6a17e84370064eec6f22cfb1717ab80cf898d82b Mon Sep 17 00:00:00 2001
-From: Nico Weber <thakis(a)chromium.org>
-Date: Tue, 10 Jan 2023 10:46:45 -0500
-Subject: [PATCH] Use python3 in all run lines
-
-We already did this in some, this converts the rest.
-Also chmod +x on write_fake_manifests.py while here.
----
- configure.py | 4 +---
- misc/measure.py | 4 +---
- misc/ninja_syntax_test.py | 2 +-
- misc/write_fake_manifests.py | 2 +-
- src/browse.py | 4 +---
- 5 files changed, 5 insertions(+), 11 deletions(-)
- mode change 100644 => 100755 misc/write_fake_manifests.py
-
-diff --git a/configure.py b/configure.py
-index 09c5b283e0..588250aa8a 100755
---- a/configure.py
-+++ b/configure.py
-@@ -1,4 +1,4 @@
--#!/usr/bin/env python
-+#!/usr/bin/env python3
- #
- # Copyright 2001 Google Inc. All Rights Reserved.
- #
-@@ -19,8 +19,6 @@
- Projects that use ninja themselves should either write a similar script
- or use a meta-build system that supports Ninja output."""
-
--from __future__ import print_function
--
- from optparse import OptionParser
- import os
- import pipes
-diff --git a/misc/measure.py b/misc/measure.py
-index 8ce95e696b..f3825efbb0 100755
---- a/misc/measure.py
-+++ b/misc/measure.py
-@@ -1,4 +1,4 @@
--#!/usr/bin/env python
-+#!/usr/bin/env python3
-
- # Copyright 2011 Google Inc. All Rights Reserved.
- #
-@@ -17,8 +17,6 @@
- """measure the runtime of a command by repeatedly running it.
- """
-
--from __future__ import print_function
--
- import time
- import subprocess
- import sys
-diff --git a/misc/ninja_syntax_test.py b/misc/ninja_syntax_test.py
-index 90ff9c6bdb..61fb177d43 100755
---- a/misc/ninja_syntax_test.py
-+++ b/misc/ninja_syntax_test.py
-@@ -1,4 +1,4 @@
--#!/usr/bin/env python
-+#!/usr/bin/env python3
-
- # Copyright 2011 Google Inc. All Rights Reserved.
- #
-diff --git a/misc/write_fake_manifests.py b/misc/write_fake_manifests.py
-old mode 100644
-new mode 100755
-index abcb677e18..bf9cf7de92
---- a/misc/write_fake_manifests.py
-+++ b/misc/write_fake_manifests.py
-@@ -1,4 +1,4 @@
--#!/usr/bin/env python
-+#!/usr/bin/env python3
-
- """Writes large manifest files, for manifest parser performance testing.
-
-diff --git a/src/browse.py b/src/browse.py
-index 653cbe91f6..b125e805a9 100755
---- a/src/browse.py
-+++ b/src/browse.py
-@@ -1,4 +1,4 @@
--#!/usr/bin/env python
-+#!/usr/bin/env python3
- #
- # Copyright 2001 Google Inc. All Rights Reserved.
- #
-@@ -20,8 +20,6 @@
- it when needed.
- """
-
--from __future__ import print_function
--
- try:
- import http.server as httpserver
- import socketserver
=====================================
projects/python-zstandard/README.md
=====================================
@@ -0,0 +1,4 @@
+Mozilla uses the python-zstandard module for various tasks in their CI, but
+they do not vendor it in Firefox's source tree.
+
+Since we build our own Python on Linux, we also need to build this module.
=====================================
projects/python-zstandard/build
=====================================
@@ -0,0 +1,14 @@
+#!/bin/bash
+[% c("var/set_default_env") -%]
+outdir=[% dest_dir _ '/' _ c('filename') %]
+mkdir -p /var/tmp/dist $outdir
+
+tar -C /var/tmp/dist -xf $rootdir/[% c('input_files_by_name/python') %]
+export PATH=/var/tmp/dist/python/bin:$PATH
+
+python3 -m pip install *.whl
+
+tar -xf [% project %]-[% c('version') %].tar.[% c('compress_tar') %]
+cd [% project %]-[% c('version') %]
+python3 setup.py bdist_wheel
+mv dist/*.whl $outdir/
=====================================
projects/python-zstandard/config
=====================================
@@ -0,0 +1,19 @@
+version: 0.24.0
+filename: 'python-zstandard-[% c("version") %]-[% c("var/build_id") %]'
+git_url: https://github.com/indygreg/python-zstandard.git
+git_hash: 9223924b1db8f36cf1c7c2dcd55232093890d145 # 0.24.0
+container:
+ use_container: 1
+
+input_files:
+ - project: container-image
+ - project: python
+ name: python
+ # This contains only binaries for Windows.
+ - name: setuptools
+ URL: https://files.pythonhosted.org/packages/a3/dc/17031897dae0efacfea57dfd3a82f…
+ sha256sum: 062d34222ad13e0cc312a4c02d73f059e86a4acbfbdea8f8f76b28c99f306922
+ # This contains only Python.
+ - name: packaging
+ URL: https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895…
+ sha256sum: 29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484
=====================================
projects/python/README.md
=====================================
@@ -17,3 +17,17 @@ image.
`browser` is a notable exception: we redefine `var/deps` for all platforms and
already add `python3` there.
+
+## OpenSSL
+
+Some Python module complain about the OpenSSL version of the container being
+too old. Therefore, we also build OpenSSL in this project, and other projects
+needing it must add `/var/tmp/dist/python/lib` to `LD_LIBRARY_PATH`.
+
+We do it here instead of using the `openssl` project because we do not want to
+rebuild a big part of the toolchain for each OpenSSL update (the module would
+be used mostly for HTTP requests, which will not go through in our builds,
+since they happen offline).
+
+When updating to a newever version of Debian for Linux containers, we might
+stop building OpenSSL and go back to using the system library.
=====================================
projects/python/build
=====================================
@@ -6,14 +6,28 @@
[% END -%]
distdir=/var/tmp/dist/[% project %]
-mkdir -p $distdir
+openssldir=/var/tmp/dist/openssl
+mkdir -p $distdir $openssldir
+
+tar -xf $rootdir/[% c('input_files_by_name/openssl') %]
+pushd openssl-3.5.4
+./Configure --prefix=$openssldir --libdir=lib
+make -j[% c("num_procs") %]
+make -j[% c("num_procs") %] install
+# Python will try to load OpenSSL during its build process.
+export LD_LIBRARY_PATH=$openssldir/lib:$LD_LIBRARY_PATH
+popd
+
tar xf [% c('input_files_by_name/python') %]
cd Python-[% c('version') %]
-./configure --prefix=$distdir --enable-optimizations
+./configure --prefix=$distdir --enable-optimizations --with-openssl=$openssldir
make -j[% c("num_procs") %]
make prefix=$distdir install
+
+cp -a $openssldir/lib/lib*.so* $distdir/lib/
+
cd /var/tmp/dist
[% c('tar', {
- tar_src => 'python',
- tar_args => '-caf ' _ dest_dir _ '/' _ c('filename'),
- }) %]
+ tar_src => 'python',
+ tar_args => '-caf ' _ dest_dir _ '/' _ c('filename'),
+ }) %]
=====================================
projects/python/config
=====================================
@@ -1,6 +1,6 @@
# vim: filetype=yaml sw=2
-version: 3.9.20
-filename: 'python-[% c("var/build_id") %].tar.[% c("compress_tar") %]'
+version: 3.13.11
+filename: 'python-[% c("version") %]-[% c("var/build_id") %].tar.[% c("compress_tar") %]'
container:
use_container: 1
var:
@@ -12,7 +12,6 @@ var:
- libffi-dev
- libncurses-dev
- libsqlite3-dev
- - libssl-dev
- zlib1g-dev
setup: |
@@ -24,7 +23,14 @@ input_files:
- project: container-image
- name: python
URL: 'https://www.python.org/ftp/python/[% c("version") %]/Python-[% c("version") %].tar.xz'
- sha256sum: 6b281279efd85294d2d6993e173983a57464c0133956fbbb5536ec9646beaf0c
+ sha256sum: 16ede7bb7cdbfa895d11b0642fa0e523f291e6487194d53cf6d3b338c3a17ea2
- name: '[% c("var/compiler") %]'
project: '[% c("var/compiler") %]'
enable: '[% c("var/linux") %]'
+ # We do not use the same we use as tor dependency because we do not want to
+ # rebuild Clang, Rust and other projects at each OpenSSL update.
+ # FWIW, these OpenSSL would be used for requests that will not go through,
+ # as our builds run offline.
+ - name: openssl
+ URL: 'https://github.com/openssl/openssl/releases/download/openssl-3.5.4/openssl-…'
+ sha256sum: 967311f84955316969bdb1d8d4b983718ef42338639c621ec4c34fddef355e99
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser-build/-/compare/…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser-build/-/compare/…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/tor-browser][base-browser-146.0a1-16.0-2] 3 commits: fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
11 Dec '25
Pier Angelo Vendrame pushed to branch base-browser-146.0a1-16.0-2 at The Tor Project / Applications / Tor Browser
Commits:
5bfdfd0b by june wilde at 2025-12-11T12:13:24+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
- - - - -
e6e81938 by june wilde at 2025-12-11T12:13:25+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 1)
- - - - -
f12904c2 by june wilde at 2025-12-11T12:13:25+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 5)
- - - - -
6 changed files:
- build/moz.configure/windows.configure
- dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
- third_party/libwebrtc/modules/desktop_capture/win/window_capture_utils.h
- third_party/libwebrtc/rtc_base/cpu_info.cc
- third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
Changes:
=====================================
build/moz.configure/windows.configure
=====================================
@@ -624,12 +624,13 @@ with only_when(depends(c_compiler)(lambda c: c.type == "clang-cl")):
add_linker_flag("-LARGEADDRESSAWARE")
add_linker_flag("-SAFESEH")
- # avoid conficts with std::min/max
- set_define("NOMINMAX", True)
-
set_define("WIN32_LEAN_AND_MEAN", True)
+with only_when(depends(c_compiler)(lambda c: c.type == "clang-cl")):
+ # See http://support.microsoft.com/kb/143208 to use STL
+ set_define("NOMINMAX", True)
+
with only_when(target_is_windows & depends(c_compiler)(lambda c: c.type != "clang-cl")):
# strsafe.h on mingw uses macros for function deprecation that pollutes namespace
=====================================
dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
=====================================
@@ -7,11 +7,11 @@
#ifndef DOM_MEDIA_WEBRTC_LIBWEBRTCOVERRIDES_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_TYPES_H_
#define DOM_MEDIA_WEBRTC_LIBWEBRTCOVERRIDES_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_TYPES_H_
-// pid_t
-#if !defined(XP_WIN) || defined(__MINGW32__)
+#if defined(XP_WIN) && \
+ !defined(__MINGW32__) // Moving this into the global namespace
+typedef int pid_t; // matching what used to be in
+#elif defined(XP_WIN) // video_capture_defines.h
# include <sys/types.h>
-#else
-typedef int pid_t;
#endif
#include "../../third_party/libwebrtc/modules/desktop_capture/desktop_capture_types.h"
=====================================
third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
=====================================
@@ -13,6 +13,8 @@
#include <dispatcherqueue.h>
#include <windows.graphics.capture.interop.h>
#include <windows.graphics.directx.direct3d11.interop.h>
+#include <windows.graphics.h>
+#include <wrl/client.h>
#include <wrl/event.h>
#include <algorithm>
=====================================
third_party/libwebrtc/modules/desktop_capture/win/window_capture_utils.h
=====================================
@@ -11,7 +11,7 @@
#ifndef MODULES_DESKTOP_CAPTURE_WIN_WINDOW_CAPTURE_UTILS_H_
#define MODULES_DESKTOP_CAPTURE_WIN_WINDOW_CAPTURE_UTILS_H_
-#include <shlobj_core.h>
+#include <shlobj.h>
#include <windows.h>
#include <wrl/client.h>
=====================================
third_party/libwebrtc/rtc_base/cpu_info.cc
=====================================
@@ -97,7 +97,7 @@ uint64_t xgetbv(uint32_t xcr) {
}
#endif // WEBRTC_ENABLE_AVX2
-#ifndef _MSC_VER
+#if !defined(_MSC_VER) && !defined(__MINGW32__)
// Intrinsic for "cpuid".
#if defined(__pic__) && defined(__i386__)
static inline void __cpuid(int cpu_info[4], int info_type) {
=====================================
third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
=====================================
@@ -12,9 +12,8 @@
#define RTC_BASE_WIN_CREATE_DIRECT3D_DEVICE_H_
#include <windows.graphics.directx.direct3d11.h>
-#ifndef __MINGW32__
-# include <windows.graphics.directX.direct3d11.interop.h>
-#else
+#include <windows.graphics.directx.direct3d11.interop.h>
+#ifdef __MINGW32__
# include <dxgi.h>
# include <inspectable.h>
extern "C" {
@@ -23,6 +22,7 @@ HRESULT __stdcall CreateDirect3D11DeviceFromDXGIDevice(
::IDXGIDevice* dxgiDevice, ::IInspectable** graphicsDevice);
}
#endif
+
#include <winerror.h>
#include <wrl/client.h>
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/e2a7c7…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/e2a7c7…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/tor-browser][tor-browser-146.0a1-16.0-2] 3 commits: fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
11 Dec '25
Pier Angelo Vendrame pushed to branch tor-browser-146.0a1-16.0-2 at The Tor Project / Applications / Tor Browser
Commits:
54d4d5c6 by june wilde at 2025-12-11T12:11:56+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
- - - - -
186f0f1a by june wilde at 2025-12-11T12:12:04+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 1)
- - - - -
9596a569 by june wilde at 2025-12-11T12:12:05+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 5)
- - - - -
6 changed files:
- build/moz.configure/windows.configure
- dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
- third_party/libwebrtc/modules/desktop_capture/win/window_capture_utils.h
- third_party/libwebrtc/rtc_base/cpu_info.cc
- third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
Changes:
=====================================
build/moz.configure/windows.configure
=====================================
@@ -624,12 +624,13 @@ with only_when(depends(c_compiler)(lambda c: c.type == "clang-cl")):
add_linker_flag("-LARGEADDRESSAWARE")
add_linker_flag("-SAFESEH")
- # avoid conficts with std::min/max
- set_define("NOMINMAX", True)
-
set_define("WIN32_LEAN_AND_MEAN", True)
+with only_when(depends(c_compiler)(lambda c: c.type == "clang-cl")):
+ # See http://support.microsoft.com/kb/143208 to use STL
+ set_define("NOMINMAX", True)
+
with only_when(target_is_windows & depends(c_compiler)(lambda c: c.type != "clang-cl")):
# strsafe.h on mingw uses macros for function deprecation that pollutes namespace
=====================================
dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
=====================================
@@ -7,11 +7,11 @@
#ifndef DOM_MEDIA_WEBRTC_LIBWEBRTCOVERRIDES_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_TYPES_H_
#define DOM_MEDIA_WEBRTC_LIBWEBRTCOVERRIDES_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_TYPES_H_
-// pid_t
-#if !defined(XP_WIN) || defined(__MINGW32__)
+#if defined(XP_WIN) && \
+ !defined(__MINGW32__) // Moving this into the global namespace
+typedef int pid_t; // matching what used to be in
+#elif defined(XP_WIN) // video_capture_defines.h
# include <sys/types.h>
-#else
-typedef int pid_t;
#endif
#include "../../third_party/libwebrtc/modules/desktop_capture/desktop_capture_types.h"
=====================================
third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
=====================================
@@ -13,6 +13,8 @@
#include <dispatcherqueue.h>
#include <windows.graphics.capture.interop.h>
#include <windows.graphics.directx.direct3d11.interop.h>
+#include <windows.graphics.h>
+#include <wrl/client.h>
#include <wrl/event.h>
#include <algorithm>
=====================================
third_party/libwebrtc/modules/desktop_capture/win/window_capture_utils.h
=====================================
@@ -11,7 +11,7 @@
#ifndef MODULES_DESKTOP_CAPTURE_WIN_WINDOW_CAPTURE_UTILS_H_
#define MODULES_DESKTOP_CAPTURE_WIN_WINDOW_CAPTURE_UTILS_H_
-#include <shlobj_core.h>
+#include <shlobj.h>
#include <windows.h>
#include <wrl/client.h>
=====================================
third_party/libwebrtc/rtc_base/cpu_info.cc
=====================================
@@ -97,7 +97,7 @@ uint64_t xgetbv(uint32_t xcr) {
}
#endif // WEBRTC_ENABLE_AVX2
-#ifndef _MSC_VER
+#if !defined(_MSC_VER) && !defined(__MINGW32__)
// Intrinsic for "cpuid".
#if defined(__pic__) && defined(__i386__)
static inline void __cpuid(int cpu_info[4], int info_type) {
=====================================
third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
=====================================
@@ -12,9 +12,8 @@
#define RTC_BASE_WIN_CREATE_DIRECT3D_DEVICE_H_
#include <windows.graphics.directx.direct3d11.h>
-#ifndef __MINGW32__
-# include <windows.graphics.directX.direct3d11.interop.h>
-#else
+#include <windows.graphics.directx.direct3d11.interop.h>
+#ifdef __MINGW32__
# include <dxgi.h>
# include <inspectable.h>
extern "C" {
@@ -23,6 +22,7 @@ HRESULT __stdcall CreateDirect3D11DeviceFromDXGIDevice(
::IDXGIDevice* dxgiDevice, ::IInspectable** graphicsDevice);
}
#endif
+
#include <winerror.h>
#include <wrl/client.h>
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/773d4e…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/compare/773d4e…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/mullvad-browser][mullvad-browser-146.0a1-16.0-2] 4 commits: MB 479 - Fix CXXFlags for libwebrtc in 146
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 11 Dec '25
11 Dec '25
Pier Angelo Vendrame pushed to branch mullvad-browser-146.0a1-16.0-2 at The Tor Project / Applications / Mullvad Browser
Commits:
ec5e89ec by june wilde at 2025-12-11T12:04:25+01:00
MB 479 - Fix CXXFlags for libwebrtc in 146
This patch will no longer be necessary in Gecko 147
- - - - -
5401bb7f by june wilde at 2025-12-11T12:04:25+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
- - - - -
7a3f13c0 by june wilde at 2025-12-11T12:04:26+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 1)
- - - - -
dcc28544 by june wilde at 2025-12-11T12:04:26+01:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 5)
- - - - -
355 changed files:
- build/moz.configure/windows.configure
- dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
- third_party/libwebrtc/BUILD.gn
- third_party/libwebrtc/api/adaptation/resource_adaptation_api_gn/moz.build
- third_party/libwebrtc/api/audio/aec3_config_gn/moz.build
- third_party/libwebrtc/api/audio/aec3_factory_gn/moz.build
- third_party/libwebrtc/api/audio/audio_frame_api_gn/moz.build
- third_party/libwebrtc/api/audio/audio_processing_gn/moz.build
- third_party/libwebrtc/api/audio/audio_processing_statistics_gn/moz.build
- third_party/libwebrtc/api/audio/builtin_audio_processing_builder_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/L16/audio_decoder_L16_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/L16/audio_encoder_L16_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/audio_codecs_api_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/builtin_audio_decoder_factory_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/builtin_audio_encoder_factory_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/g711/audio_decoder_g711_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/g711/audio_encoder_g711_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/g722/audio_decoder_g722_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/g722/audio_encoder_g722_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus/audio_decoder_multiopus_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus/audio_decoder_opus_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus/audio_encoder_multiopus_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus/audio_encoder_opus_config_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus/audio_encoder_opus_gn/moz.build
- third_party/libwebrtc/api/audio_codecs/opus_audio_decoder_factory_gn/moz.build
- third_party/libwebrtc/api/audio_options_api_gn/moz.build
- third_party/libwebrtc/api/crypto/options_gn/moz.build
- third_party/libwebrtc/api/environment/environment_factory_gn/moz.build
- third_party/libwebrtc/api/field_trials_registry_gn/moz.build
- third_party/libwebrtc/api/frame_transformer_factory_gn/moz.build
- third_party/libwebrtc/api/frame_transformer_interface_gn/moz.build
- third_party/libwebrtc/api/media_stream_interface_gn/moz.build
- third_party/libwebrtc/api/neteq/default_neteq_controller_factory_gn/moz.build
- third_party/libwebrtc/api/neteq/default_neteq_factory_gn/moz.build
- third_party/libwebrtc/api/neteq/neteq_api_gn/moz.build
- third_party/libwebrtc/api/neteq/tick_timer_gn/moz.build
- third_party/libwebrtc/api/priority_gn/moz.build
- third_party/libwebrtc/api/rtc_error_gn/moz.build
- third_party/libwebrtc/api/rtc_event_log/rtc_event_log_gn/moz.build
- third_party/libwebrtc/api/rtp_headers_gn/moz.build
- third_party/libwebrtc/api/rtp_packet_info_gn/moz.build
- third_party/libwebrtc/api/rtp_parameters_gn/moz.build
- third_party/libwebrtc/api/rtp_sender_setparameters_callback_gn/moz.build
- third_party/libwebrtc/api/task_queue/pending_task_safety_flag_gn/moz.build
- third_party/libwebrtc/api/task_queue/task_queue_gn/moz.build
- third_party/libwebrtc/api/transport/bitrate_settings_gn/moz.build
- third_party/libwebrtc/api/transport/field_trial_based_config_gn/moz.build
- third_party/libwebrtc/api/transport/goog_cc_gn/moz.build
- third_party/libwebrtc/api/transport/network_control_gn/moz.build
- third_party/libwebrtc/api/transport/rtp/dependency_descriptor_gn/moz.build
- third_party/libwebrtc/api/transport_api_gn/moz.build
- third_party/libwebrtc/api/units/data_rate_gn/moz.build
- third_party/libwebrtc/api/units/data_size_gn/moz.build
- third_party/libwebrtc/api/units/frequency_gn/moz.build
- third_party/libwebrtc/api/units/time_delta_gn/moz.build
- third_party/libwebrtc/api/units/timestamp_gn/moz.build
- third_party/libwebrtc/api/video/builtin_video_bitrate_allocator_factory_gn/moz.build
- third_party/libwebrtc/api/video/encoded_frame_gn/moz.build
- third_party/libwebrtc/api/video/encoded_image_gn/moz.build
- third_party/libwebrtc/api/video/frame_buffer_gn/moz.build
- third_party/libwebrtc/api/video/video_adaptation_gn/moz.build
- third_party/libwebrtc/api/video/video_bitrate_allocation_gn/moz.build
- third_party/libwebrtc/api/video/video_bitrate_allocator_gn/moz.build
- third_party/libwebrtc/api/video/video_frame_gn/moz.build
- third_party/libwebrtc/api/video/video_frame_i010_gn/moz.build
- third_party/libwebrtc/api/video/video_frame_metadata_gn/moz.build
- third_party/libwebrtc/api/video/video_rtp_headers_gn/moz.build
- third_party/libwebrtc/api/video_codecs/builtin_video_decoder_factory_gn/moz.build
- third_party/libwebrtc/api/video_codecs/rtc_software_fallback_wrappers_gn/moz.build
- third_party/libwebrtc/api/video_codecs/scalability_mode_gn/moz.build
- third_party/libwebrtc/api/video_codecs/video_codecs_api_gn/moz.build
- third_party/libwebrtc/api/video_codecs/vp8_temporal_layers_factory_gn/moz.build
- third_party/libwebrtc/audio/audio_gn/moz.build
- third_party/libwebrtc/audio/utility/audio_frame_operations_gn/moz.build
- third_party/libwebrtc/call/adaptation/resource_adaptation_gn/moz.build
- third_party/libwebrtc/call/bitrate_allocator_gn/moz.build
- third_party/libwebrtc/call/bitrate_configurator_gn/moz.build
- third_party/libwebrtc/call/call_gn/moz.build
- third_party/libwebrtc/call/call_interfaces_gn/moz.build
- third_party/libwebrtc/call/payload_type_picker_gn/moz.build
- third_party/libwebrtc/call/rtp_interfaces_gn/moz.build
- third_party/libwebrtc/call/rtp_receiver_gn/moz.build
- third_party/libwebrtc/call/rtp_sender_gn/moz.build
- third_party/libwebrtc/call/version_gn/moz.build
- third_party/libwebrtc/call/video_receive_stream_api_gn/moz.build
- third_party/libwebrtc/call/video_send_stream_api_gn/moz.build
- third_party/libwebrtc/common_audio/common_audio_avx2_gn/moz.build
- third_party/libwebrtc/common_audio/common_audio_cc_gn/moz.build
- third_party/libwebrtc/common_audio/common_audio_gn/moz.build
- third_party/libwebrtc/common_audio/common_audio_neon_gn/moz.build
- third_party/libwebrtc/common_audio/common_audio_sse2_gn/moz.build
- third_party/libwebrtc/common_audio/fir_filter_factory_gn/moz.build
- third_party/libwebrtc/common_audio/third_party/ooura/fft_size_128_gn/moz.build
- third_party/libwebrtc/common_audio/third_party/ooura/fft_size_256_gn/moz.build
- third_party/libwebrtc/common_video/common_video_gn/moz.build
- third_party/libwebrtc/common_video/corruption_detection_converters_gn/moz.build
- third_party/libwebrtc/common_video/generic_frame_descriptor/generic_frame_descriptor_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_audio_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_bwe_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_field_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_number_encodings_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_pacing_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_rtp_rtcp_gn/moz.build
- third_party/libwebrtc/logging/rtc_event_video_gn/moz.build
- third_party/libwebrtc/logging/rtc_stream_config_gn/moz.build
- third_party/libwebrtc/media/adapted_video_track_source_gn/moz.build
- third_party/libwebrtc/media/codec_gn/moz.build
- third_party/libwebrtc/media/media_constants_gn/moz.build
- third_party/libwebrtc/media/rid_description_gn/moz.build
- third_party/libwebrtc/media/rtc_audio_video_gn/moz.build
- third_party/libwebrtc/media/rtc_internal_video_codecs_gn/moz.build
- third_party/libwebrtc/media/rtc_sdp_video_format_utils_gn/moz.build
- third_party/libwebrtc/media/rtc_simulcast_encoder_adapter_gn/moz.build
- third_party/libwebrtc/media/video_adapter_gn/moz.build
- third_party/libwebrtc/media/video_broadcaster_gn/moz.build
- third_party/libwebrtc/media/video_common_gn/moz.build
- third_party/libwebrtc/media/video_source_base_gn/moz.build
- third_party/libwebrtc/modules/async_audio_processing/async_audio_processing_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/audio_coding_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/audio_coding_opus_common_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/audio_encoder_cng_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/audio_network_adaptor_config_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/audio_network_adaptor_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/g711_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/g722_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/legacy_encoded_audio_frame_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/neteq_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/pcm16b_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/red_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/webrtc_cng_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/webrtc_multiopus_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/webrtc_opus_gn/moz.build
- third_party/libwebrtc/modules/audio_coding/webrtc_opus_wrapper_gn/moz.build
- third_party/libwebrtc/modules/audio_mixer/audio_frame_manipulator_gn/moz.build
- third_party/libwebrtc/modules/audio_mixer/audio_mixer_impl_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/aec3/aec3_avx2_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/aec3/aec3_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/aec_dump/null_aec_dump_factory_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/aec_dump_interface_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/aecm/aecm_core_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc/agc_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc/legacy_agc_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc/level_estimation_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/adaptive_digital_gain_controller_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/biquad_filter_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/clipping_predictor_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/cpu_features_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/fixed_digital_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/gain_applier_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/input_volume_controller_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/input_volume_stats_reporter_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/noise_level_estimator_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_auto_correlation_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_layers_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_lp_residual_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_pitch_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/rnn_vad_spectral_features_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/rnn_vad/vector_math_avx2_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/saturation_protector_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/speech_level_estimator_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/agc2/vad_wrapper_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/apm_logging_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/audio_buffer_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/audio_frame_proxies_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/audio_processing_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/capture_levels_adjuster/capture_levels_adjuster_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/gain_controller2_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/high_pass_filter_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/ns/ns_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/post_filter_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/rms_level_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/utility/cascaded_biquad_filter_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/utility/legacy_delay_estimator_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/utility/pffft_wrapper_gn/moz.build
- third_party/libwebrtc/modules/audio_processing/vad/vad_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/congestion_controller_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/alr_detector_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/delay_based_bwe_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/estimators_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/goog_cc_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/link_capacity_estimator_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/loss_based_bwe_v2_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/probe_controller_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/pushback_controller_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/goog_cc/send_side_bwe_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/rtp/control_handler_gn/moz.build
- third_party/libwebrtc/modules/congestion_controller/rtp/transport_feedback_gn/moz.build
- third_party/libwebrtc/modules/desktop_capture/desktop_capture_differ_sse2_gn/moz.build
- third_party/libwebrtc/modules/desktop_capture/desktop_capture_gn/moz.build
- third_party/libwebrtc/modules/desktop_capture/primitives_gn/moz.build
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
- third_party/libwebrtc/modules/desktop_capture/win/window_capture_utils.h
- third_party/libwebrtc/modules/pacing/interval_budget_gn/moz.build
- third_party/libwebrtc/modules/pacing/pacing_gn/moz.build
- third_party/libwebrtc/modules/remote_bitrate_estimator/congestion_control_feedback_generator_gn/moz.build
- third_party/libwebrtc/modules/remote_bitrate_estimator/remote_bitrate_estimator_gn/moz.build
- third_party/libwebrtc/modules/remote_bitrate_estimator/transport_sequence_number_feedback_generator_gn/moz.build
- third_party/libwebrtc/modules/rtp_rtcp/leb128_gn/moz.build
- third_party/libwebrtc/modules/rtp_rtcp/ntp_time_util_gn/moz.build
- third_party/libwebrtc/modules/rtp_rtcp/rtp_rtcp_format_gn/moz.build
- third_party/libwebrtc/modules/rtp_rtcp/rtp_rtcp_gn/moz.build
- third_party/libwebrtc/modules/rtp_rtcp/rtp_video_header_gn/moz.build
- third_party/libwebrtc/modules/video_capture/video_capture_internal_impl_gn/moz.build
- third_party/libwebrtc/modules/video_capture/video_capture_module_gn/moz.build
- third_party/libwebrtc/modules/video_coding/chain_diff_calculator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/codecs/av1/av1_svc_config_gn/moz.build
- third_party/libwebrtc/modules/video_coding/codecs/av1/dav1d_decoder_gn/moz.build
- third_party/libwebrtc/modules/video_coding/codecs/av1/libaom_av1_encoder_gn/moz.build
- third_party/libwebrtc/modules/video_coding/encoded_frame_gn/moz.build
- third_party/libwebrtc/modules/video_coding/frame_dependencies_calculator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/frame_helpers_gn/moz.build
- third_party/libwebrtc/modules/video_coding/h264_sprop_parameter_sets_gn/moz.build
- third_party/libwebrtc/modules/video_coding/h26x_packet_buffer_gn/moz.build
- third_party/libwebrtc/modules/video_coding/nack_requester_gn/moz.build
- third_party/libwebrtc/modules/video_coding/packet_buffer_gn/moz.build
- third_party/libwebrtc/modules/video_coding/svc/scalability_mode_util_gn/moz.build
- third_party/libwebrtc/modules/video_coding/svc/scalability_structures_gn/moz.build
- third_party/libwebrtc/modules/video_coding/svc/scalable_video_controller_gn/moz.build
- third_party/libwebrtc/modules/video_coding/svc/simulcast_to_svc_converter_gn/moz.build
- third_party/libwebrtc/modules/video_coding/svc/svc_rate_allocator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/decode_time_percentile_filter_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/frame_delay_variation_kalman_filter_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/inter_frame_delay_variation_calculator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/jitter_estimator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/rtt_filter_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/timestamp_extrapolator_gn/moz.build
- third_party/libwebrtc/modules/video_coding/timing/timing_module_gn/moz.build
- third_party/libwebrtc/modules/video_coding/video_codec_interface_gn/moz.build
- third_party/libwebrtc/modules/video_coding/video_coding_gn/moz.build
- third_party/libwebrtc/modules/video_coding/video_coding_utility_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_h264_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_libvpx_interface_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_vp8_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_vp8_scalability_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_vp8_temporal_layers_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_vp9_gn/moz.build
- third_party/libwebrtc/modules/video_coding/webrtc_vp9_helpers_gn/moz.build
- third_party/libwebrtc/moz-patch-stack/s0129.patch
- third_party/libwebrtc/net/dcsctp/packet/chunk_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/chunk_validators_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/crc32c_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/error_cause_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/parameter_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/sctp_packet_gn/moz.build
- third_party/libwebrtc/net/dcsctp/packet/tlv_trait_gn/moz.build
- third_party/libwebrtc/net/dcsctp/public/factory_gn/moz.build
- third_party/libwebrtc/net/dcsctp/public/socket_gn/moz.build
- third_party/libwebrtc/net/dcsctp/rx/data_tracker_gn/moz.build
- third_party/libwebrtc/net/dcsctp/rx/interleaved_reassembly_streams_gn/moz.build
- third_party/libwebrtc/net/dcsctp/rx/reassembly_queue_gn/moz.build
- third_party/libwebrtc/net/dcsctp/rx/traditional_reassembly_streams_gn/moz.build
- third_party/libwebrtc/net/dcsctp/socket/dcsctp_socket_gn/moz.build
- third_party/libwebrtc/net/dcsctp/socket/heartbeat_handler_gn/moz.build
- third_party/libwebrtc/net/dcsctp/socket/packet_sender_gn/moz.build
- third_party/libwebrtc/net/dcsctp/socket/stream_reset_handler_gn/moz.build
- third_party/libwebrtc/net/dcsctp/socket/transmission_control_block_gn/moz.build
- third_party/libwebrtc/net/dcsctp/timer/task_queue_timeout_gn/moz.build
- third_party/libwebrtc/net/dcsctp/timer/timer_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/outstanding_data_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/retransmission_error_counter_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/retransmission_queue_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/retransmission_timeout_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/rr_send_queue_gn/moz.build
- third_party/libwebrtc/net/dcsctp/tx/stream_scheduler_gn/moz.build
- third_party/libwebrtc/rtc_base/async_dns_resolver_gn/moz.build
- third_party/libwebrtc/rtc_base/base64_gn/moz.build
- third_party/libwebrtc/rtc_base/bit_buffer_gn/moz.build
- third_party/libwebrtc/rtc_base/bitrate_tracker_gn/moz.build
- third_party/libwebrtc/rtc_base/bitstream_reader_gn/moz.build
- third_party/libwebrtc/rtc_base/byte_buffer_gn/moz.build
- third_party/libwebrtc/rtc_base/checks_gn/moz.build
- third_party/libwebrtc/rtc_base/containers/flat_containers_internal_gn/moz.build
- third_party/libwebrtc/rtc_base/copy_on_write_buffer_gn/moz.build
- third_party/libwebrtc/rtc_base/cpu_info.cc
- third_party/libwebrtc/rtc_base/cpu_info_gn/moz.build
- third_party/libwebrtc/rtc_base/criticalsection_gn/moz.build
- third_party/libwebrtc/rtc_base/denormal_disabler_gn/moz.build
- third_party/libwebrtc/rtc_base/event_tracer_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/alr_experiment_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/balanced_degradation_settings_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/encoder_info_settings_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/field_trial_parser_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/keyframe_interval_settings_experiment_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/min_video_bitrate_experiment_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/normalize_simulcast_size_experiment_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/quality_scaler_settings_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/quality_scaling_experiment_gn/moz.build
- third_party/libwebrtc/rtc_base/experiments/rate_control_settings_gn/moz.build
- third_party/libwebrtc/rtc_base/frequency_tracker_gn/moz.build
- third_party/libwebrtc/rtc_base/histogram_percentile_counter_gn/moz.build
- third_party/libwebrtc/rtc_base/ip_address_gn/moz.build
- third_party/libwebrtc/rtc_base/logging_gn/moz.build
- third_party/libwebrtc/rtc_base/memory/aligned_malloc_gn/moz.build
- third_party/libwebrtc/rtc_base/net_helpers_gn/moz.build
- third_party/libwebrtc/rtc_base/network/sent_packet_gn/moz.build
- third_party/libwebrtc/rtc_base/network_constants_gn/moz.build
- third_party/libwebrtc/rtc_base/network_route_gn/moz.build
- third_party/libwebrtc/rtc_base/null_socket_server_gn/moz.build
- third_party/libwebrtc/rtc_base/platform_thread_gn/moz.build
- third_party/libwebrtc/rtc_base/platform_thread_types_gn/moz.build
- third_party/libwebrtc/rtc_base/race_checker_gn/moz.build
- third_party/libwebrtc/rtc_base/random_gn/moz.build
- third_party/libwebrtc/rtc_base/rate_limiter_gn/moz.build
- third_party/libwebrtc/rtc_base/rate_statistics_gn/moz.build
- third_party/libwebrtc/rtc_base/rate_tracker_gn/moz.build
- third_party/libwebrtc/rtc_base/rtc_event_gn/moz.build
- third_party/libwebrtc/rtc_base/rtc_numerics_gn/moz.build
- third_party/libwebrtc/rtc_base/rtp_to_ntp_estimator_gn/moz.build
- third_party/libwebrtc/rtc_base/sample_counter_gn/moz.build
- third_party/libwebrtc/rtc_base/socket_address_gn/moz.build
- third_party/libwebrtc/rtc_base/socket_gn/moz.build
- third_party/libwebrtc/rtc_base/stringutils_gn/moz.build
- third_party/libwebrtc/rtc_base/synchronization/sequence_checker_internal_gn/moz.build
- third_party/libwebrtc/rtc_base/synchronization/yield_gn/moz.build
- third_party/libwebrtc/rtc_base/synchronization/yield_policy_gn/moz.build
- third_party/libwebrtc/rtc_base/system/file_wrapper_gn/moz.build
- third_party/libwebrtc/rtc_base/task_utils/repeating_task_gn/moz.build
- third_party/libwebrtc/rtc_base/third_party/sigslot/sigslot_gn/moz.build
- third_party/libwebrtc/rtc_base/threading_gn/moz.build
- third_party/libwebrtc/rtc_base/timeutils_gn/moz.build
- third_party/libwebrtc/rtc_base/weak_ptr_gn/moz.build
- third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
- third_party/libwebrtc/rtc_base/win/create_direct3d_device_gn/moz.build
- third_party/libwebrtc/rtc_base/win/get_activation_factory_gn/moz.build
- third_party/libwebrtc/rtc_base/win/hstring_gn/moz.build
- third_party/libwebrtc/rtc_base/win/windows_version_gn/moz.build
- third_party/libwebrtc/rtc_base/win32_gn/moz.build
- third_party/libwebrtc/rtc_base/zero_memory_gn/moz.build
- third_party/libwebrtc/system_wrappers/field_trial_gn/moz.build
- third_party/libwebrtc/system_wrappers/metrics_gn/moz.build
- third_party/libwebrtc/system_wrappers/system_wrappers_gn/moz.build
- third_party/libwebrtc/video/adaptation/video_adaptation_gn/moz.build
- third_party/libwebrtc/video/config/encoder_config_gn/moz.build
- third_party/libwebrtc/video/config/streams_config_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/corruption_classifier_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/frame_instrumentation_evaluation_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/frame_instrumentation_generator_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/generic_mapping_functions_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/halton_frame_sampler_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/halton_sequence_gn/moz.build
- third_party/libwebrtc/video/corruption_detection/video_frame_sampler_gn/moz.build
- third_party/libwebrtc/video/decode_synchronizer_gn/moz.build
- third_party/libwebrtc/video/frame_cadence_adapter_gn/moz.build
- third_party/libwebrtc/video/frame_decode_timing_gn/moz.build
- third_party/libwebrtc/video/frame_dumping_decoder_gn/moz.build
- third_party/libwebrtc/video/frame_dumping_encoder_gn/moz.build
- third_party/libwebrtc/video/render/incoming_video_stream_gn/moz.build
- third_party/libwebrtc/video/render/video_render_frames_gn/moz.build
- third_party/libwebrtc/video/task_queue_frame_decode_scheduler_gn/moz.build
- third_party/libwebrtc/video/unique_timestamp_counter_gn/moz.build
- third_party/libwebrtc/video/video_gn/moz.build
- third_party/libwebrtc/video/video_receive_stream_timeout_tracker_gn/moz.build
- third_party/libwebrtc/video/video_stream_buffer_controller_gn/moz.build
- third_party/libwebrtc/video/video_stream_encoder_impl_gn/moz.build
The diff was not included because it is too large.
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/f2…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/f2…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/tor-browser][tor-browser-146.0a1-16.0-2] fixup! TB 40597: Implement TorSettings module
by Pier Angelo Vendrame (ï¼ pierov) 10 Dec '25
by Pier Angelo Vendrame (ï¼ pierov) 10 Dec '25
10 Dec '25
Pier Angelo Vendrame pushed to branch tor-browser-146.0a1-16.0-2 at The Tor Project / Applications / Tor Browser
Commits:
773d4ea2 by Beatriz Rizental at 2025-12-10T18:59:48-03:00
fixup! TB 40597: Implement TorSettings module
Fix linter issue.
- - - - -
1 changed file:
- toolkit/content/pt_config.json
Changes:
=====================================
toolkit/content/pt_config.json
=====================================
@@ -25,4 +25,3 @@
]
}
}
-
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/commit/773d4ea…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/tor-browser/-/commit/773d4ea…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/mullvad-browser][mullvad-browser-146.0a1-16.0-2] 26 commits: Add CI for Mullvad Browser
by brizental (ï¼ brizental) 10 Dec '25
by brizental (ï¼ brizental) 10 Dec '25
10 Dec '25
brizental pushed to branch mullvad-browser-146.0a1-16.0-2 at The Tor Project / Applications / Mullvad Browser
Commits:
85adecf4 by Henry Wilkes at 2025-12-10T16:39:36-03:00
Add CI for Mullvad Browser
- - - - -
e29a4001 by Pier Angelo Vendrame at 2025-12-10T16:39:37-03:00
MB 38: Mullvad Browser configuration
- - - - -
69e248f9 by Pier Angelo Vendrame at 2025-12-10T16:39:38-03:00
MB 1: Mullvad Browser branding
See also:
mullvad-browser#5: Product name and directory customization
mullvad-browser#12: Create new branding directories and integrate Mullvad icons+branding
mullvad-browser#14: Remove Default Built-in bookmarks
mullvad-browser#35: Add custom PDF icons for Windows builds
mullvad-browser#48: Replace Mozilla copyright and legal trademarks in mullvadbrowser.exe metadata
mullvad-browser#51: Update trademark string
mullvad-browser#104: Update shipped dll metadata copyright/licensing info
mullvad-browser#107: Add alpha and nightly icons
- - - - -
c73a63d6 by Henry Wilkes at 2025-12-10T16:39:38-03:00
Mullvad Browser strings
This commit adds strings needed by the following Mullvad Browser
patches.
- - - - -
a3a66322 by Pier Angelo Vendrame at 2025-12-10T16:39:39-03:00
MB 20: Allow packaged-addons in PBM.
We install a few addons from the distribution directory, but they are
not automatically enabled for PBM mode.
This commit modifies the code that installs them to also add the PBM
permission to the known ones.
- - - - -
a955fdb4 by Pier Angelo Vendrame at 2025-12-10T16:39:40-03:00
MB 63: Customize some about pages for Mullvad Browser
Also:
mullvad-browser#57: Purge unneeded about: pages
- - - - -
2badb349 by Pier Angelo Vendrame at 2025-12-10T16:39:41-03:00
MB 37: Customization for the about dialog
- - - - -
f3dc8161 by Henry Wilkes at 2025-12-10T16:39:41-03:00
MB 39: Add home page about:mullvad-browser
- - - - -
99b04d9f by Henry Wilkes at 2025-12-10T16:39:42-03:00
fixup! MB 39: Add home page about:mullvad-browser
MB 486: Delay the update data for preloaded about:mullvad-browser pages.
- - - - -
3862b7bb by hackademix at 2025-12-10T16:39:43-03:00
MB 97: Remove UI cues to install new extensions.
- - - - -
e011b26d by hackademix at 2025-12-10T16:39:43-03:00
MB 47: uBlock Origin customization
- - - - -
db657613 by Pier Angelo Vendrame at 2025-12-10T16:39:44-03:00
MB 21: Disable the password manager
This commit disables the about:login page and removes the "Login and
Password" section of about:preferences.
We do not do anything to the real password manager of Firefox, that is
in toolkit: it contains C++ parts that make it difficult to actually
prevent it from being built..
Finally, we modify the the function that opens about:login to report an
error in the console so that we can quickly get a backtrace to the code
that tries to use it.
- - - - -
64a15fba by Pier Angelo Vendrame at 2025-12-10T16:39:45-03:00
MB 112: Updater customization for Mullvad Browser
MB 71: Set the updater base URL to Mullvad domain
- - - - -
32f5edf3 by Nicolas Vigier at 2025-12-10T16:39:45-03:00
MB 79: Add Mullvad Browser MAR signing keys
MB 256: Add mullvad-browser nightly mar signing key
- - - - -
edab066e by Pier Angelo Vendrame at 2025-12-10T16:39:46-03:00
MB 34: Hide unsafe and unwanted preferences UI
about:preferences allow to override some of our defaults, that could
be fingeprintable or have some other unwanted consequences.
- - - - -
e29f5a3c by Pier Angelo Vendrame at 2025-12-10T16:39:47-03:00
MB 160: Disable the cookie exceptions button
Besides disabling the "Delete on close checkbox", disable also the
"Manage Exceptions" button when always using PBM.
- - - - -
db6083fc by hackademix at 2025-12-10T16:39:48-03:00
MB 163: Prevent uBlock Origin from being uninstalled/disabled
- - - - -
73ccf78e by Richard Pospesel at 2025-12-10T17:12:16-03:00
MB 188: Customize Gitlab Issue and Merge templates
- - - - -
3944d4ca by rui hildt at 2025-12-10T17:24:06-03:00
MB 213: Customize the search engines list.
- - - - -
b2432b25 by Pier Angelo Vendrame at 2025-12-10T17:24:07-03:00
fixup! MB 213: Customize the search engines list.
MB 487: Use custom order for search engines.
- - - - -
8732aec0 by Henry Wilkes at 2025-12-10T17:24:08-03:00
fixup! MB 213: Customize the search engines list.
MB 488: Drop Mullvad Leta search engine.
- - - - -
2ff0cbf9 by hackademix at 2025-12-10T17:24:08-03:00
MB 214: Enable cross-tab identity leak protection in "quiet" mode
- - - - -
0f238f5a by Pier Angelo Vendrame at 2025-12-10T17:24:09-03:00
MB 320: Temporarily disable WebRTC and WDBA on Windows.
WebRTC should be re-enabled when tor-browser#42758 is resolved, and and
the default browser agent when in general we make this feature work
again.
- - - - -
1c5f7922 by Henry Wilkes at 2025-12-10T17:24:10-03:00
MB 329: Customize toolbar for mullvad-browser.
- - - - -
e1f10fbf by Henry Wilkes at 2025-12-10T17:24:10-03:00
MB 419: Mullvad Browser migration procedures.
This commit implements the the Mullvad Browser's version of _migrateUI.
- - - - -
f20c93e3 by Henry Wilkes at 2025-12-10T17:24:11-03:00
MB 488: Adjust search engine removal notification for Mullvad Leta.
- - - - -
242 changed files:
- .gitlab/ci/jobs/update-translations.yml
- .gitlab/issue_templates/000 Bug Report.md
- .gitlab/issue_templates/010 Proposal.md
- .gitlab/issue_templates/020 Web Compatibility.md
- .gitlab/issue_templates/030 Test.md
- .gitlab/issue_templates/040 Feature.md
- .gitlab/issue_templates/060 Rebase - Alpha.md
- .gitlab/issue_templates/061 Rebase - Stable.md
- .gitlab/issue_templates/063 Rebase - Rapid.md
- .gitlab/issue_templates/090 Emergency Security Issue.md
- .gitlab/merge_request_templates/Default.md
- + .gitlab/merge_request_templates/Rebase.md
- browser/app/Makefile.in
- browser/app/macbuild/Contents/Info.plist.in
- browser/app/module.ver
- browser/app/firefox.exe.manifest → browser/app/mullvadbrowser.exe.manifest
- + browser/app/profile/000-mullvad-browser.js
- browser/app/profile/001-base-profile.js
- browser/base/content/aboutDialog.xhtml
- browser/base/content/appmenu-viewcache.inc.xhtml
- browser/base/content/browser-menubar.inc
- browser/base/content/browser-places.js
- browser/base/content/browser.js
- browser/base/content/default-bookmarks.html
- browser/base/content/nsContextMenu.sys.mjs
- browser/base/content/overrides/app-license.html
- browser/base/content/pageinfo/pageInfo.xhtml
- browser/base/content/utilityOverlay.js
- browser/branding/branding-common.mozbuild
- + browser/branding/mb-alpha/VisualElements_150.png
- + browser/branding/mb-alpha/VisualElements_70.png
- + browser/branding/mb-alpha/configure.sh
- + browser/branding/mb-alpha/content/about-logo.png
- + browser/branding/mb-alpha/content/about-logo.svg
- + browser/branding/mb-alpha/content/about-logo(a)2x.png
- + browser/branding/mb-alpha/content/about-wordmark.svg
- + browser/branding/mb-alpha/content/about.png
- + browser/branding/mb-alpha/content/aboutDialog.css
- + browser/branding/mb-alpha/content/document_pdf.svg
- + browser/branding/mb-alpha/content/firefox-wordmark.svg
- + browser/branding/mb-alpha/content/icon128.png
- + browser/branding/mb-alpha/content/icon16.png
- + browser/branding/mb-alpha/content/icon256.png
- + browser/branding/mb-alpha/content/icon32.png
- + browser/branding/mb-alpha/content/icon48.png
- + browser/branding/mb-alpha/content/icon64.png
- + browser/branding/mb-alpha/content/jar.mn
- + browser/branding/mb-alpha/content/moz.build
- + browser/branding/mb-alpha/content/mullvad-branding.css
- + browser/branding/mb-alpha/default128.png
- + browser/branding/mb-alpha/default16.png
- + browser/branding/mb-alpha/default22.png
- + browser/branding/mb-alpha/default24.png
- + browser/branding/mb-alpha/default256.png
- + browser/branding/mb-alpha/default32.png
- + browser/branding/mb-alpha/default48.png
- + browser/branding/mb-alpha/default64.png
- + browser/branding/mb-alpha/document.icns
- + browser/branding/mb-alpha/document.ico
- + browser/branding/mb-alpha/document_pdf.ico
- + browser/branding/mb-alpha/firefox.icns
- + browser/branding/mb-alpha/firefox.ico
- + browser/branding/mb-alpha/firefox.svg
- + browser/branding/mb-alpha/locales/en-US/brand.ftl
- + browser/branding/mb-alpha/locales/en-US/brand.properties
- + browser/branding/mb-alpha/locales/jar.mn
- + browser/branding/mb-alpha/locales/moz.build
- + browser/branding/mb-alpha/locales/mullvad-about-wordmark-en.ftl
- + browser/branding/mb-alpha/moz.build
- + browser/branding/mb-alpha/mullvadbrowser.VisualElementsManifest.xml
- + browser/branding/mb-alpha/newtab.ico
- + browser/branding/mb-alpha/newwindow.ico
- + browser/branding/mb-alpha/pbmode.ico
- + browser/branding/mb-alpha/pref/firefox-branding.js
- + browser/branding/mb-nightly/VisualElements_150.png
- + browser/branding/mb-nightly/VisualElements_70.png
- + browser/branding/mb-nightly/configure.sh
- + browser/branding/mb-nightly/content/about-logo.png
- + browser/branding/mb-nightly/content/about-logo.svg
- + browser/branding/mb-nightly/content/about-logo(a)2x.png
- + browser/branding/mb-nightly/content/about-wordmark.svg
- + browser/branding/mb-nightly/content/about.png
- + browser/branding/mb-nightly/content/aboutDialog.css
- + browser/branding/mb-nightly/content/document_pdf.svg
- + browser/branding/mb-nightly/content/firefox-wordmark.svg
- + browser/branding/mb-nightly/content/icon128.png
- + browser/branding/mb-nightly/content/icon16.png
- + browser/branding/mb-nightly/content/icon256.png
- + browser/branding/mb-nightly/content/icon32.png
- + browser/branding/mb-nightly/content/icon48.png
- + browser/branding/mb-nightly/content/icon64.png
- + browser/branding/mb-nightly/content/jar.mn
- + browser/branding/mb-nightly/content/moz.build
- + browser/branding/mb-nightly/content/mullvad-branding.css
- + browser/branding/mb-nightly/default128.png
- + browser/branding/mb-nightly/default16.png
- + browser/branding/mb-nightly/default22.png
- + browser/branding/mb-nightly/default24.png
- + browser/branding/mb-nightly/default256.png
- + browser/branding/mb-nightly/default32.png
- + browser/branding/mb-nightly/default48.png
- + browser/branding/mb-nightly/default64.png
- + browser/branding/mb-nightly/document.icns
- + browser/branding/mb-nightly/document.ico
- + browser/branding/mb-nightly/document_pdf.ico
- + browser/branding/mb-nightly/firefox.icns
- + browser/branding/mb-nightly/firefox.ico
- + browser/branding/mb-nightly/firefox.svg
- + browser/branding/mb-nightly/locales/en-US/brand.ftl
- + browser/branding/mb-nightly/locales/en-US/brand.properties
- + browser/branding/mb-nightly/locales/jar.mn
- + browser/branding/mb-nightly/locales/moz.build
- + browser/branding/mb-nightly/locales/mullvad-about-wordmark-en.ftl
- + browser/branding/mb-nightly/moz.build
- + browser/branding/mb-nightly/mullvadbrowser.VisualElementsManifest.xml
- + browser/branding/mb-nightly/newtab.ico
- + browser/branding/mb-nightly/newwindow.ico
- + browser/branding/mb-nightly/pbmode.ico
- + browser/branding/mb-nightly/pref/firefox-branding.js
- + browser/branding/mb-release/VisualElements_150.png
- + browser/branding/mb-release/VisualElements_70.png
- + browser/branding/mb-release/configure.sh
- + browser/branding/mb-release/content/about-logo.png
- + browser/branding/mb-release/content/about-logo.svg
- + browser/branding/mb-release/content/about-logo(a)2x.png
- + browser/branding/mb-release/content/about-wordmark.svg
- + browser/branding/mb-release/content/about.png
- + browser/branding/mb-release/content/aboutDialog.css
- + browser/branding/mb-release/content/document_pdf.svg
- + browser/branding/mb-release/content/firefox-wordmark.svg
- + browser/branding/mb-release/content/icon128.png
- + browser/branding/mb-release/content/icon16.png
- + browser/branding/mb-release/content/icon256.png
- + browser/branding/mb-release/content/icon32.png
- + browser/branding/mb-release/content/icon48.png
- + browser/branding/mb-release/content/icon64.png
- + browser/branding/mb-release/content/jar.mn
- + browser/branding/mb-release/content/moz.build
- + browser/branding/mb-release/content/mullvad-branding.css
- + browser/branding/mb-release/default128.png
- + browser/branding/mb-release/default16.png
- + browser/branding/mb-release/default22.png
- + browser/branding/mb-release/default24.png
- + browser/branding/mb-release/default256.png
- + browser/branding/mb-release/default32.png
- + browser/branding/mb-release/default48.png
- + browser/branding/mb-release/default64.png
- + browser/branding/mb-release/document.icns
- + browser/branding/mb-release/document.ico
- + browser/branding/mb-release/document_pdf.ico
- + browser/branding/mb-release/firefox.icns
- + browser/branding/mb-release/firefox.ico
- + browser/branding/mb-release/firefox.svg
- + browser/branding/mb-release/locales/en-US/brand.ftl
- + browser/branding/mb-release/locales/en-US/brand.properties
- + browser/branding/mb-release/locales/jar.mn
- + browser/branding/mb-release/locales/moz.build
- + browser/branding/mb-release/locales/mullvad-about-wordmark-en.ftl
- + browser/branding/mb-release/moz.build
- + browser/branding/mb-release/mullvadbrowser.VisualElementsManifest.xml
- + browser/branding/mb-release/newtab.ico
- + browser/branding/mb-release/newwindow.ico
- + browser/branding/mb-release/pbmode.ico
- + browser/branding/mb-release/pref/firefox-branding.js
- browser/components/BrowserContentHandler.sys.mjs
- browser/components/BrowserGlue.sys.mjs
- browser/components/DesktopActorRegistry.sys.mjs
- browser/components/ProfileDataUpgrader.sys.mjs
- browser/components/about/AboutRedirector.cpp
- browser/components/about/components.conf
- browser/components/customizableui/CustomizableUI.sys.mjs
- browser/components/moz.build
- + browser/components/mullvad-browser/AboutMullvadBrowserChild.sys.mjs
- + browser/components/mullvad-browser/AboutMullvadBrowserParent.sys.mjs
- + browser/components/mullvad-browser/content/2728-sparkles.svg
- + browser/components/mullvad-browser/content/aboutMullvadBrowser.css
- + browser/components/mullvad-browser/content/aboutMullvadBrowser.html
- + browser/components/mullvad-browser/content/aboutMullvadBrowser.js
- + browser/components/mullvad-browser/jar.mn
- + browser/components/mullvad-browser/moz.build
- browser/components/preferences/home.inc.xhtml
- browser/components/preferences/preferences.xhtml
- browser/components/preferences/privacy.inc.xhtml
- browser/components/preferences/privacy.js
- browser/components/preferences/search.inc.xhtml
- browser/components/search/SearchUIUtils.sys.mjs
- browser/components/tabbrowser/NewTabPagePreloading.sys.mjs
- browser/config/mozconfigs/base-browser
- + browser/config/mozconfigs/mullvad-browser
- browser/installer/package-manifest.in
- browser/installer/windows/nsis/updater_append.ini
- browser/locales/l10n.toml
- browser/modules/HomePage.sys.mjs
- browser/moz.build
- browser/moz.configure
- config/create_rc.py
- devtools/client/aboutdebugging/src/actions/runtimes.js
- devtools/client/aboutdebugging/src/components/sidebar/Sidebar.js
- devtools/client/jar.mn
- devtools/client/themes/images/aboutdebugging-firefox-aurora.svg
- devtools/client/themes/images/aboutdebugging-firefox-beta.svg
- devtools/client/themes/images/aboutdebugging-firefox-logo.svg
- devtools/client/themes/images/aboutdebugging-firefox-nightly.svg
- devtools/client/themes/images/aboutdebugging-firefox-release.svg
- + devtools/client/themes/images/aboutdebugging-mullvadbrowser-logo.svg
- docshell/base/nsAboutRedirector.cpp
- docshell/build/components.conf
- moz.configure
- mozconfig-linux-aarch64
- mozconfig-linux-aarch64-dev
- mozconfig-linux-x86_64
- mozconfig-linux-x86_64-asan
- mozconfig-linux-x86_64-dev
- mozconfig-macos
- mozconfig-macos-dev
- mozconfig-windows-x86_64
- toolkit/components/extensions/child/ext-storage.js
- toolkit/components/extensions/parent/ext-storage.js
- toolkit/components/passwordmgr/LoginHelper.sys.mjs
- toolkit/components/search/SearchService.sys.mjs
- toolkit/components/search/content/base-browser-search-engine-icons.json
- toolkit/components/search/content/base-browser-search-engines.json
- + toolkit/components/search/content/brave.svg
- + toolkit/components/search/content/mojeek.ico
- toolkit/components/search/tests/xpcshell/test_base_browser.js
- toolkit/components/securitylevel/SecurityLevel.sys.mjs
- + toolkit/content/aboutRightsMullvad.xhtml
- + toolkit/content/aboutTelemetryMullvad.xhtml
- toolkit/content/jar.mn
- + toolkit/locales/en-US/toolkit/global/mullvad-browser.ftl
- toolkit/mozapps/extensions/AddonManager.sys.mjs
- toolkit/mozapps/extensions/content/aboutaddons.css
- toolkit/mozapps/extensions/internal/XPIDatabase.sys.mjs
- toolkit/mozapps/extensions/internal/XPIProvider.sys.mjs
- toolkit/mozapps/update/updater/nightly_aurora_level3_primary.der
- toolkit/mozapps/update/updater/nightly_aurora_level3_secondary.der
- toolkit/mozapps/update/updater/release_primary.der
- toolkit/mozapps/update/updater/release_secondary.der
- + toolkit/themes/shared/icons/mullvadbrowser.png
- toolkit/themes/shared/minimal-toolkit.jar.inc.mn
- toolkit/xre/nsAppRunner.cpp
- tools/lint/fluent-lint/exclusions.yml
The diff was not included because it is too large.
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/e2…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/e2…
You're receiving this email because of your account on gitlab.torproject.org.
1
0
[Git][tpo/applications/mullvad-browser][mullvad-browser-146.0a1-16.0-2] 138 commits: BB 41454: Move focus after calling openPreferences for a sub-category.
by brizental (ï¼ brizental) 10 Dec '25
by brizental (ï¼ brizental) 10 Dec '25
10 Dec '25
brizental pushed to branch mullvad-browser-146.0a1-16.0-2 at The Tor Project / Applications / Mullvad Browser
Commits:
12b70da7 by Henry Wilkes at 2025-11-27T14:10:04-03:00
BB 41454: Move focus after calling openPreferences for a sub-category.
Temporary fix until mozilla bug 1799153 gets a patch upstream.
- - - - -
6dd21530 by Henry Wilkes at 2025-11-27T14:17:06-03:00
BB 43072: Add aria label and description to moz-message-bar.
Ensures that moz-message-bar, including notifications, are announced on
Orca.
This addresses upstream bugzilla bug 1895857 and should likely be
replaced when it is fixed.
- - - - -
0742ec69 by Henry Wilkes at 2025-11-27T14:17:14-03:00
BB 42739: Use the brand name for profile error messages.
Some messages in profileSelection.properties use gAppData->name as
variable inputs. However, gAppData->name is still "Firefox" for our
base-browser builds, rather than the user-facing browser name. We swap
these instances with the displayed brand name instead.
- - - - -
70c86106 by Beatriz Rizental at 2025-12-01T10:18:54-03:00
fixup! BB 42739: Use the brand name for profile error messages.
- - - - -
9cdfd142 by Henry Wilkes at 2025-12-01T10:18:55-03:00
BB 41483: Remove the firefox override for appstrings.properties
Remove this patch after upstream bugzilla bug 1790187
- - - - -
f0e459de by Sukhbir Singh at 2025-12-01T10:18:56-03:00
BB 44167: Don't build the uninstaller for Windows during Firefox compilation
- - - - -
910a17f9 by Marco Simonelli at 2025-12-01T10:18:57-03:00
BB 41459: WebRTC fails to build under mingw (Part 1)
- properly define NOMINMAX for just MSVC builds
- - - - -
faff4a6f by Marco Simonelli at 2025-12-01T10:18:57-03:00
BB 41459: WebRTC fails to build under mingw (Part 2)
- fixes required to build third_party/libwebrtc
- - - - -
48ceade7 by Pier Angelo Vendrame at 2025-12-01T10:18:58-03:00
fixup! BB 41459: WebRTC fails to build under mingw (Part 2)
Linted.
- - - - -
ebf6263a by Marco Simonelli at 2025-12-01T10:18:59-03:00
BB 41459: WebRTC fails to build under mingw (Part 3)
- fixes required to build third_party/sipcc
- - - - -
4804326e by Marco Simonelli at 2025-12-01T10:19:00-03:00
BB 41459: WebRTC fails to build under mingw (Part 4)
- fixes requried to build netwerk/sctp
- - - - -
1df69286 by Marco Simonelli at 2025-12-01T10:19:01-03:00
BB 41459: WebRTC fails to build under mingw (Part 5)
- fixes required to build dom/media/webrtc
- - - - -
3df5124e by Marco Simonelli at 2025-12-01T10:19:01-03:00
BB 41459: WebRTC fails to build under mingw (Part 6)
- fixes required to build dom/media/systemservices
- - - - -
9ed6aa55 by june wilde at 2025-12-01T10:19:02-03:00
BB 42758: Fix WebRTC build errors.
- - - - -
803a78a6 by hackademix at 2025-12-01T10:19:03-03:00
BB 41854: Allow overriding download spam protection.
- - - - -
bbb5d632 by hackademix at 2025-12-01T10:19:04-03:00
BB 42832: Download spam prevention exemption for browser extensions.
- - - - -
7207589f by Pier Angelo Vendrame at 2025-12-01T10:19:04-03:00
BB 42220: Allow for more file types to be forced-inline.
Firefox allows to open some files in the browser without any
confirmation, but this will result in a disk leak, because the file will
be downloaded to the temporary directory first (and not deleted, in some
cases).
A preference allows PDFs to be opened without being downloaded to disk.
So, we introduce a similar one to do the same for all the files that are
set to be opened automatically in the browser, except svg and html files
to prevent XSS hazards (see BB 43211).
- - - - -
93ce669d by hackademix at 2025-12-01T10:19:05-03:00
BB 42835: Create an actor to filter file data transfers
- - - - -
1d243ba8 by Pier Angelo Vendrame at 2025-12-01T10:19:06-03:00
BB 44103: Remove ref-names from attr's .git_archival.txt.
The currently vendored copy of python-attrs's .git_archival.txt
includes a ref-names field which might make source tarballs not
reproducible.
Upstream fixed this in da2333cd37747d692d5c78b0c94bd400ff883a9a,
therefore we apply the same change until Mozilla vendors an updated
copy of python-attrs.
See also https://bugzilla.mozilla.org/show_bug.cgi?id=1980103.
- - - - -
445c589e by hackademix at 2025-12-01T10:19:07-03:00
BB 44125: Do not offer to save signatures by default in Private Browsing Mode
- - - - -
29e654e6 by Beatriz Rizental at 2025-12-09T20:14:59-03:00
BB 43564: Modify ./mach bootstrap for Base Browser
- - - - -
0be03b8a by Beatriz Rizental at 2025-12-09T20:15:00-03:00
fixup! BB 43564: Modify ./mach bootstrap for Base Browser
- - - - -
2d5dcce0 by Beatriz Rizental at 2025-12-09T20:15:00-03:00
BB 42728: Modify ./mach lint to skip unused linters
- - - - -
9ac30957 by Beatriz Rizental at 2025-12-09T20:15:01-03:00
fixup! BB 42728: Modify ./mach lint to skip unused linters
- - - - -
a5151db3 by Beatriz Rizental at 2025-12-09T20:15:02-03:00
fixup! BB 42728: Modify ./mach lint to skip unused linters
- - - - -
8916d972 by Morgan at 2025-12-10T13:59:01-03:00
BB 43615: Add Gitlab Issue and Merge Request templates
- - - - -
dcce69a8 by Pier Angelo Vendrame at 2025-12-10T13:59:14-03:00
fixup! BB 43615: Add Gitlab Issue and Merge Request templates
Update GitLab templates.
Avoid mentioning existing issues/MRs, and update the channels.
- - - - -
e8647685 by Morgan at 2025-12-10T14:21:08-03:00
fixup! BB 43615: Add Gitlab Issue and Merge Request templates
made header formatting consistent
- - - - -
8d82db10 by Morgan at 2025-12-10T14:21:08-03:00
fixup! BB 43615: Add Gitlab Issue and Merge Request templates
drill down on more detailed Bookkeeping information
- - - - -
94c43da7 by Henry Wilkes at 2025-12-10T14:21:09-03:00
BB 41803: Add some developer tools for working on tor-browser.
- - - - -
32e28ebb by Richard Pospesel at 2025-12-10T14:21:10-03:00
BB 42683: Create script to generate issue triage csv file from bugzilla query and git logs
- - - - -
84be9572 by Henry Wilkes at 2025-12-10T14:21:10-03:00
BB 42305: Add script to combine translation files across versions.
- - - - -
14610d3e by Beatriz Rizental at 2025-12-10T14:21:11-03:00
BB 43535: Enable tests
- - - - -
2c0dc432 by Beatriz Rizental at 2025-12-10T14:21:12-03:00
Add CI for Base Browser
- - - - -
203b3db4 by Beatriz Rizental at 2025-12-10T14:21:12-03:00
fixup! Add CI for Base Browser
Use custom CI container for translations job.
- - - - -
44a4e6e4 by Beatriz Rizental at 2025-12-10T14:21:13-03:00
fixup! Add CI for Base Browser
Filed a review ticket https://gitlab.torproject.org/tpo/applications/tor-browser/-/issues/44402
Turned a linter rule back to warning for now.
- - - - -
15084ad5 by Pier Angelo Vendrame at 2025-12-10T15:26:00-03:00
Base Browser's .mozconfigs.
Bug 17858: Cannot create incremental MARs for hardened builds.
Define HOST_CFLAGS, etc. to avoid compiling programs such as mbsdiff
(which is part of mar-tools and is not distributed to end-users) with
ASan.
Bug 21849: Don't allow SSL key logging.
Bug 25741 - TBA: Disable features at compile-time
Define MOZ_ANDROID_NETWORK_STATE and MOZ_ANDROID_LOCATION
Bug 27623 - Export MOZILLA_OFFICIAL during desktop builds
This fixes a problem where some preferences had the wrong default value.
Also see bug 27472 where we made a similar fix for Android.
Bug 29859: Disable HLS support for now
Bug 30463: Explicitly disable MOZ_TELEMETRY_REPORTING
Bug 32493: Disable MOZ_SERVICES_HEALTHREPORT
Bug 33734: Set MOZ_NORMANDY to False
Bug 33851: Omit Parental Controls.
Bug 40252: Add --enable-rust-simd to our tor-browser mozconfig files
Bug 41584: Move some configuration options to base-browser level
- - - - -
b67d0469 by Pier Angelo Vendrame at 2025-12-10T15:26:00-03:00
BB 44118: Disable Mozilla's nightly and early beta features.
- - - - -
573b47a2 by Pier Angelo Vendrame at 2025-12-10T15:26:01-03:00
fixup! BB 44118: Disable Mozilla's nightly and early beta features.
It seems is_early_beta_or_earlier can be only True or None, but not
False.
- - - - -
fde8cf84 by Pier Angelo Vendrame at 2025-12-10T15:26:02-03:00
Tweaks to the build system
Bug 40857: Modified the fat .aar creation file
This is a workaround to build fat .aars with the compiling enviornment
disabled.
Mozilla does not use a similar configuration, but either runs a Firefox
build and discards its output, or uses artifacts build.
We might switch to artifact builds too, and drop this patch, or write a
better one to upstream. But until then we need this patch.
See also https://bugzilla.mozilla.org/show_bug.cgi?id=1763770.
Bug 41458: Prevent `mach package-multi-locale` from actually creating a package
macOS builds need some files to be moved around with
./mach package-multi-locale to create multi-locale packages.
The required command isn't exposed through any other mach command.
So, we patch package-multi-locale both to prevent it from failing when
doing official builds and to detect any future changes on it.
- - - - -
d0aa9093 by Pier Angelo Vendrame at 2025-12-10T15:26:03-03:00
fixup! Tweaks to the build system
Disable more build telemetry.
The sentry module uses the Python ssl module, which does not work in
our old Linux build environment.
- - - - -
08b5fea9 by Pier Angelo Vendrame at 2025-12-10T15:26:03-03:00
fixup! Tweaks to the build system
TB 44370: Always create target.maven.zip on Android.
- - - - -
4e4d1a17 by Beatriz Rizental at 2025-12-10T15:26:04-03:00
fixup! Tweaks to the build system
Fix linter issues. Related to using things that are not being imported
since we commented them out. Guess the linter doesnt care that the code
is unreacheable.
- - - - -
4468abc5 by Pier Angelo Vendrame at 2025-12-10T15:26:05-03:00
BB 29320: Replace the gnu target with gnullvm for Rust.
- - - - -
6ba932f9 by Pier Angelo Vendrame at 2025-12-10T15:26:06-03:00
BB 42616: Remove VideoCaptureTest.kt.
This is a workaround to fix the GeckoView build with WebRTC disabled.
We should replace this workaround with a proper solution, that
excludes this test when MOZ_WEBRTC is undefined/False.
- - - - -
757b820f by Pier Angelo Vendrame at 2025-12-10T15:26:07-03:00
BB 41108: Remove privileged macOS installation from 102
- - - - -
4e85e65f by Dan Ballard at 2025-12-10T15:26:07-03:00
BB 41149: Re-enable DLL injection protection in all builds not just nightlies
- - - - -
fa94035c by Henry Wilkes at 2025-12-10T15:26:08-03:00
BB 43092: Disable wayland by default in Base Browser.
- - - - -
fe62c0d1 by Matthew Finkel at 2025-12-10T15:26:09-03:00
BB 24796: Comment out excess permissions from GeckoView
The GeckoView AndroidManifest.xml is not preprocessed unlike Fennec's
manifest, so we can't use the ifdef preprocessor guards around the
permissions we do not want. Commenting the permissions is the
next-best-thing.
- - - - -
1b2e19e4 by Matthew Finkel at 2025-12-10T15:26:10-03:00
BB 28125: Prevent non-Necko network connections
- - - - -
6903f75f by Mike Perry at 2025-12-10T15:26:10-03:00
BB 12974: Disable NTLM and Negotiate HTTP Auth
The Mozilla bugs: https://bugzilla.mozilla.org/show_bug.cgi?id=1046421,
https://bugzilla.mozilla.org/show_bug.cgi?id=1261591, tor-browser#27602
- - - - -
62b0a9e0 by cypherpunks1 at 2025-12-10T15:26:11-03:00
BB 40717: Hide Windows SSO in settings
- - - - -
203ec0fb by Georg Koppen at 2025-12-10T15:26:12-03:00
BB 16285: Exclude ClearKey system for now
In the past the ClearKey system had not been compiled when specifying
--disable-eme. But that changed and it is even bundled nowadays (see:
Mozilla's bug 1300654). We don't want to ship it right now as the use
case for it is not really visible while the code had security
vulnerabilities in the past.
- - - - -
179b0231 by Kathy Brade at 2025-12-10T15:26:13-03:00
BB 21431: Clean-up system extensions shipped in Firefox
Only ship the pdfjs extension.
- - - - -
9a0ed398 by Kathy Brade at 2025-12-10T15:26:14-03:00
BB 33852: Clean up about:logins (LockWise) to avoid mentioning sync, etc.
Hide elements on about:logins that mention sync, "Firefox LockWise", and
Mozilla's LockWise mobile apps.
Disable the "Create New Login" button when security.nocertdb is true.
- - - - -
73eee8f9 by Alex Catarineu at 2025-12-10T15:26:14-03:00
BB 41457: Remove Mozilla permissions
Bug 40025: Remove Mozilla add-on install permissions
- - - - -
e8f9ff11 by Henry Wilkes at 2025-12-10T15:26:15-03:00
BB 44045: Disable ML features.
- - - - -
f4556882 by Kathy Brade at 2025-12-10T15:26:16-03:00
BB 41662: Disable about:sync-logs
Even though we disable sync by default with
`identity.fxaccounts.enabled`, this about: page is still avilable.
We could throw an exception on the constructor of the related
component, but it would result only in an error in the console, without
a visible "this address does not look right" error page.
If we fix the issues with MOZ_SERVICES_SYNC, we can restore the
component.
- - - - -
56e28e8a by Morgan at 2025-12-10T15:26:17-03:00
BB 42716: Disable unwanted about: pages
- - - - -
276cc80f by Arthur Edelstein at 2025-12-10T15:26:18-03:00
BB 26353: Prevent speculative connect that violated FPI.
Connections were observed in the catch-all circuit when
the user entered an https or http URL in the URL bar, or
typed a search term.
- - - - -
fd525e57 by Alex Catarineu at 2025-12-10T15:26:19-03:00
BB 31740: Remove some unnecessary RemoteSettings instances
More concretely, SearchService.jsm 'hijack-blocklists' and
url-classifier-skip-urls.
Avoid creating instance for 'anti-tracking-url-decoration'.
If prefs are disabling their usage, avoid creating instances for
'cert-revocations' and 'intermediates'.
Do not ship JSON dumps for collections we do not expect to need. For
the ones in the 'main' bucket, this prevents them from being synced
unnecessarily (the code in remote-settings does so for collections
in the main bucket for which a dump or local data exists). For the
collections in the other buckets, we just save some size by not
shipping their dumps.
We also clear the collections database on the v2 -> v3 migration.
- - - - -
23a89c57 by cypherpunks1 at 2025-12-10T15:26:20-03:00
BB 41092: Add a RemoteSettings JSON dump for query-stripping
- - - - -
5b11c6cf by cypherpunks1 at 2025-12-10T15:26:20-03:00
BB 42730: Patch RemoteSettings to use only local dumps as a data source
- - - - -
f012d163 by Pier Angelo Vendrame at 2025-12-10T15:26:21-03:00
BB 43525: Skip Remote Settings for search engine customization.
Also, add some bundled search engines.
- - - - -
0a701470 by Pier Angelo Vendrame at 2025-12-10T15:26:22-03:00
BB 41635: Disable the Normandy component
Do not include Normandy at all whenever MOZ_NORMANDY is False.
- - - - -
7ae38a9c by Georg Koppen at 2025-12-10T15:26:23-03:00
BB 30541: Disable WebGL readPixel() for web content
Related Bugzilla: https://bugzilla.mozilla.org/show_bug.cgi?id=1428034
- - - - -
7df665e6 by Alex Catarineu at 2025-12-10T15:26:24-03:00
BB 28369: Stop shipping pingsender executable
- - - - -
94f18b1f by Pier Angelo Vendrame at 2025-12-10T15:26:24-03:00
BB 41599: Always return an empty string as network ID
Firefox computes an internal network ID used to detect network changes
and act consequently (e.g., to improve WebSocket UX).
However, there are a few ways to get this internal network ID, so we
patch them out, to be sure any new code will not be able to use them and
possibly link users.
We also sent a patch to Mozilla to seed the internal network ID, to
prevent any accidental leak in the future.
Upstream: https://bugzilla.mozilla.org/show_bug.cgi?id=1817756
- - - - -
9c4fa354 by Pier Angelo Vendrame at 2025-12-10T15:26:25-03:00
BB 43386: Use Firefox in the UA in RFP-exempt request.
XHR requests initiated by extensions are exempt from RFP.
Therefore, they report the actual app name, instead of Firefox, and the
actual Firefox minor version.
This happens whenever the app name has been customized and does not
match a hardcoded "Firefox".
- - - - -
822cd44d by Richard Pospesel at 2025-12-10T15:26:26-03:00
BB 41327: Disable UrlbarProviderInterventions
- - - - -
65191e56 by Richard Pospesel at 2025-12-10T15:26:27-03:00
BB 42037: Disable about:firefoxview page
- - - - -
7cf1fcf0 by Henry Wilkes at 2025-12-10T15:26:28-03:00
BB 44107: Re-include firefoxview asset view-opentabs.svg.
Should be dropped after bugzilla bug 1987279 is resolved.
- - - - -
5d99777c by Mike Perry at 2025-12-10T15:26:29-03:00
Firefox preference overrides.
- - - - -
6760e001 by Pier Angelo Vendrame at 2025-12-10T15:26:29-03:00
fixup! Firefox preference overrides.
BB 44396: Fix the name of CJK fonts in font lists.
- - - - -
b41b3745 by Pier Angelo Vendrame at 2025-12-10T15:26:30-03:00
BB 43140: Tighten up fonts on Linux.
We ship a fontconfig configuration to esnure all Linux users have the
same defaults and reduce the chances of fingerprinting.
We used to add this file in tor-browser-build, but some users might
skip it by using the browser executable without the wrapper script.
Therefore, we decided to activate the file with the FontConfig API
instead of the environment variables.
Bug 41043: Hardcode the UI font on Linux
The mechanism to choose the UI font does not play well with our
fontconfig configuration. As a result, the final criterion to choose
the font for the UI was its version.
Since we hardcode Arimo as a default sans-serif on preferences, we use
it also for the UI. FontConfig will fall back to some other font for
scripts Arimo does not cover as expected (we tested with Japanese).
Bug 43141: Hardcode system-ui to Arimo.
- - - - -
cc197102 by Pier Angelo Vendrame at 2025-12-10T15:26:31-03:00
BB 43322: Customize the font visibility lists.
Customize the lists of fonts to assign base visibility to in
base browser and derivatives.
Also, rename the files with the upstream lists, to make sure we do not
use them by mistake.
- - - - -
d6a45e55 by Alex Catarineu at 2025-12-10T15:26:32-03:00
BB 30605: Honor privacy.spoof_english in Android
This checks `privacy.spoof_english` whenever `setLocales` is
called from Fenix side and sets `intl.accept_languages`
accordingly.
Bug 40198: Expose privacy.spoof_english pref in GeckoView
- - - - -
22316cfc by Pier Angelo Vendrame at 2025-12-10T15:26:33-03:00
BB 42562: Normalized the Accepted Languages on Android.
The OS language might be outside the list of actually supported
languages and it might leak the user's region.
Therefore, we force the locale reported in Accept-Language to match one
we support with translations, even when it means using a not exact
region tag.
- - - - -
f41d44df by Alex Catarineu at 2025-12-10T15:26:34-03:00
BB 40171: Make WebRequest and GeckoWebExecutor First-Party aware
- - - - -
8eaa06ef by Alex Catarineu at 2025-12-10T15:26:34-03:00
BB 26345: Hide tracking protection UI
- - - - -
4e85a48b by Henry Wilkes at 2025-12-10T15:26:35-03:00
BB 43109: Hide Firefox Relay from settings.
This should remain disabled, see tor-browser#42814.
- - - - -
e23489d7 by Henry Wilkes at 2025-12-10T15:26:36-03:00
BB 42777: Hide Website Privacy Preferences.
We hide the Website Privacy Preferences section, which controls the
"global privacy control" (GPC) and "do not track" (DNT) settings.
- - - - -
c6c4e957 by Morgan at 2025-12-10T15:26:37-03:00
BB 42070: Hide "Use smooth scrolling" from settings
- - - - -
5f706880 by Arthur Edelstein at 2025-12-10T15:26:38-03:00
BB 18905: Hide unwanted items from help menu
Bug 25660: Remove the "New Private Window" option
- - - - -
46e53ecb by Pier Angelo Vendrame at 2025-12-10T15:26:39-03:00
BB 41739: Remove "Website appearance" from about:preferences.
It is ignored because of RFP and it is confusing for users.
- - - - -
589dea8c by Henry Wilkes at 2025-12-10T15:26:39-03:00
BB 43850: Modify the Contrast Control settings for RFP.
- - - - -
70107552 by Henry Wilkes at 2025-12-10T15:26:40-03:00
BB 43117: Hide "Always underline links" from settings.
- - - - -
a13cd1b5 by Pier Angelo Vendrame at 2025-12-10T15:26:41-03:00
BB 42774: Always hide the third-pary certs UI.
- - - - -
b1dc87d9 by Henry Wilkes at 2025-12-10T15:26:42-03:00
BB 43118: Hide feature recommendation (CFR) settings.
- - - - -
4a1161f6 by Henry Wilkes at 2025-12-10T15:26:43-03:00
BB 44279: Disable contextual search install prompt.
- - - - -
5668c3b2 by Pier Angelo Vendrame at 2025-12-10T15:26:44-03:00
BB 9173: Change the default Firefox profile directory to be relative.
This commit makes Firefox look for the default profile directory in a
directory relative to the binary path.
The directory can be specified through the --with-relative-data-dir.
This is relative to the same directory as the firefox main binary for
Linux and Windows.
On macOS, we remove Contents/MacOS from it.
Or, in other words, the directory is relative to the application
bundle.
This behavior can be overriden at runtime, by placing a file called
system-install adjacent to the firefox main binary (also on macOS).
- - - - -
3871f53c by Pier Angelo Vendrame at 2025-12-10T15:26:44-03:00
BB 42773: Replace ~ with the original home.
In Bug 93141, Mozilla started sending users to their home when they type
~ in the URL bar.
On Linux, we change $HOME for various reason, therefore you would be
redirected to the spoofed home directory when typing ~.
So, we check if the original home directory is known, and use that,
instead.
- - - - -
233fc4c3 by Alex Catarineu at 2025-12-10T15:26:45-03:00
BB 27604: Fix addon issues when moving the profile directory
Bugzilla: https://bugzilla.mozilla.org/show_bug.cgi?id=1429838
- - - - -
1d5bff49 by Mike Perry at 2025-12-10T15:26:46-03:00
BB 13028: Prevent potential proxy bypass cases.
It looks like these cases should only be invoked in the NSS command line
tools, and not the browser, but I decided to patch them anyway because there
literally is a maze of network function pointers being passed around, and it's
very hard to tell if some random code might not pass in the proper proxied
versions of the networking code here by accident.
Bugzilla: https://bugzilla.mozilla.org/show_bug.cgi?id=1433509
- - - - -
fb1a09ab by Pier Angelo Vendrame at 2025-12-10T15:26:47-03:00
BB 40309: Avoid using regional OS locales
Avoid regional OS locales if the pref
`intl.regional_prefs.use_os_locales` is false but RFP is enabled.
- - - - -
cf8171dc by Matthew Finkel at 2025-12-10T15:26:48-03:00
BB 40432: Prevent probing installed applications
Bugzilla: https://bugzilla.mozilla.org/show_bug.cgi?id=1711084
- - - - -
eb3e0db4 by Henry Wilkes at 2025-12-10T15:26:49-03:00
BB 29745: Limit remote access to content accessible resources
- - - - -
c98b0a99 by cypherpunks1 at 2025-12-10T15:26:50-03:00
BB 33955: When copying an image only copy the image contents to the clipboard
- - - - -
88c50497 by cypherpunks1 at 2025-12-10T15:26:50-03:00
BB 41791: Omit the source URL when copying page contents to the clipboard
- - - - -
e08e3649 by Pier Angelo Vendrame at 2025-12-10T15:26:51-03:00
BB 43196: Remove the vendor name from media notifications on Linux.
Firefox shows "vendor remoteName" as a title of the "... is playing
media" notification on Linux.
However, for our browser the remote name is enough, and prepending the
vendor to it creates a string users usually never see.
- - - - -
585f872a by Pier Angelo Vendrame at 2025-12-10T15:26:52-03:00
Base Browser strings
This commit adds all the strings needed by following Base Browser
patches.
- - - - -
3b531650 by Henry Wilkes at 2025-12-10T15:26:53-03:00
BB 42583: Modify moz-support-link for Base Browser.
- - - - -
c5d6bc93 by Pier Angelo Vendrame at 2025-12-10T15:26:54-03:00
BB 41369: Improve Firefox language settings for multi-lingual packages
Change the language selector to be sorted by language code, rather than
name, and to display the language code to the user.
Bug 41372: Handle Japanese as a special case in preferences on macOS
Japanese is treated in a special way on macOS. However, seeing the
Japanese language tag could be confusing for users, and moreover the
language name is not localized correctly like other langs.
Bug 41378: Tell users that they can change their language at the first start
With multi-lingual builds, Tor Browser matches the user's system
language, but some users might want to change it.
So, we tell them that it is possible, but only once.
- - - - -
7c04f31b by p13dz at 2025-12-10T15:26:55-03:00
BB 40283: Workaround for the file upload bug
- - - - -
9fc2664f by hackademix at 2025-12-10T15:26:55-03:00
BB 42019: Empty browser's clipboard on browser shutdown
- - - - -
06621e7d by hackademix at 2025-12-10T15:26:56-03:00
BB 42084: Ensure English spoofing works even if preferences are set out of order.
- - - - -
e02cd821 by Pier Angelo Vendrame at 2025-12-10T15:26:57-03:00
BB 41930: Remove the UI to customize accept_languages.
- - - - -
488ee9f7 by hackademix at 2025-12-10T15:26:58-03:00
BB 32308: Use direct browser sizing for letterboxing.
Bug 30556: align letterboxing with 200x100 new win width stepping
- - - - -
3c1e6fab by Henry Wilkes at 2025-12-10T15:26:59-03:00
fixup! BB 32308: Use direct browser sizing for letterboxing.
TB 44365: Fix CSS linting errors and missing variables.
- - - - -
eec3955a by hackademix at 2025-12-10T15:26:59-03:00
BB 41631: Prevent weird initial window dimensions caused by subpixel computations
- - - - -
85aee552 by hackademix at 2025-12-10T15:27:00-03:00
BB 41918: Option to reuse last window size when letterboxing is enabled.
- - - - -
61c325a8 by hackademix at 2025-12-10T15:27:01-03:00
BB 41916: Letterboxing preferences UI
- - - - -
0b911cee by hackademix at 2025-12-10T15:27:02-03:00
BB 41695: Warn on window maximization without letterboxing in RFPHelper module
- - - - -
ee258624 by hackademix at 2025-12-10T15:27:03-03:00
BB 42443: Shrink window to match letterboxing size when the emtpy area is clicked.
- - - - -
c2c6ebc1 by hackademix at 2025-12-10T15:27:04-03:00
BB 41919: Letterboxing, add temporarily visible web content-size indicator on window resizing.
- - - - -
9160ba71 by Henry Wilkes at 2025-12-10T15:27:04-03:00
fixup! BB 41919: Letterboxing, add temporarily visible web content-size indicator on window resizing.
TB 44214: Fix letterboxing status indicator for RTL.
- - - - -
68d8c7f3 by Henry Wilkes at 2025-12-10T15:27:05-03:00
BB 42528: Don't leak system scrollbar size on windows.
- - - - -
a4c95884 by Henry Wilkes at 2025-12-10T15:27:06-03:00
BB 31575: Disable Firefox Home (Activity Stream)
Treat about:blank as the default home page and new tab page.
BB 43886: Disable the newtab component/addon.
Bug 41624: Disable about:pocket-* pages.
Bug 40144: Redirect about:privatebrowsing to the user's home
- - - - -
ca509ad0 by Pier Angelo Vendrame at 2025-12-10T15:27:07-03:00
fixup! BB 31575: Disable Firefox Home (Activity Stream)
BB 44302: Add URI_CAN_LOAD_IN_PRIVILEGEDABOUT_PROCESS to some about pages.
- - - - -
5c6c36af by Kathy Brade at 2025-12-10T15:27:07-03:00
BB 4234: Use the Firefox Update Process for Base Browser.
Windows: disable "runas" code path in updater (15201).
Windows: avoid writing to the registry (16236).
Also includes fixes for tickets 13047, 13301, 13356, 13594, 15406,
16014, 16909, 24476, and 25909.
Also fix bug 27221: purge the startup cache if the Base Browser
version changed (even if the Firefox version and build ID did
not change), e.g., after a minor Base Browser update.
Also fix 32616: Disable GetSecureOutputDirectoryPath() functionality.
Bug 26048: potentially confusing "restart to update" message
Within the update doorhanger, remove the misleading message that mentions
that windows will be restored after an update is applied, and replace the
"Restart and Restore" button label with an existing
"Restart to update Tor Browser" string.
Bug 28885: notify users that update is downloading
Add a "Downloading Base Browser update" item which appears in the
hamburger (app) menu while the update service is downloading a MAR
file. Before this change, the browser did not indicate to the user
that an update was in progress, which is especially confusing in
Tor Browser because downloads often take some time. If the user
clicks on the new menu item, the about dialog is opened to allow
the user to see download progress.
As part of this fix, the update service was changed to always show
update-related messages in the hamburger menu, even if the update
was started in the foreground via the about dialog or via the
"Check for Tor Browser Update" toolbar menu item. This change is
consistent with the Tor Browser goal of making sure users are
informed about the update process.
Removed #28885 parts of this patch which have been uplifted to Firefox.
- - - - -
009a8cb6 by Pier Angelo Vendrame at 2025-12-10T15:27:08-03:00
BB 42061: Create an alpha update channel.
- - - - -
2b45ecb9 by Nicolas Vigier at 2025-12-10T15:27:09-03:00
BB 41682: Add base-browser nightly mar signing key
- - - - -
3f3f7c02 by Pier Angelo Vendrame at 2025-12-10T15:27:10-03:00
BB 41603: Customize the creation of MOZ_SOURCE_URL
MOZ_SOURCE_URL is created by combining MOZ_SOURCE_REPO and
MOZ_SOURCE_CHANGESET.
But the code takes for granted that it refers to a Hg instance, so it
combines them as `$MOZ_SOURCE_REPO/rev/$MOZ_SOURCE_CHANGESET`.
With this commit, we change this logic to combine them to create a URL
that is valid for GitLab.
$MOZ_SOURCE_CHANGESET needs to be a commit hash, not a branch or a tag.
If that is needed, we could use /-/tree/, instead of /-/commit/.
- - - - -
aede2f33 by Pier Angelo Vendrame at 2025-12-10T15:27:10-03:00
BB 42438: Tweaks to the migration wizard.
Remove the items not compatible with our features (such as history) from
the migration wizard.
On Linux, allow to specify an alternative home directory, since we
usually change $HOME in our startup script.
- - - - -
a9f0eba2 by Alex Catarineu at 2025-12-10T15:27:11-03:00
BB 40069: Add helpers for message passing with extensions
- - - - -
b978ebc1 by Matthew Finkel at 2025-12-10T15:27:12-03:00
BB 41598: Prevent NoScript from being removed/disabled.
Bug 40253: Explicitly allow NoScript in Private Browsing mode.
- - - - -
90311b2d by Henry Wilkes at 2025-12-10T15:27:13-03:00
BB 41581: Hide NoScript extension's toolbar button by default.
This hides it from both the toolbar and the unified extensions panel.
We also hide the unified-extension-button if the panel would be empty:
not including the NoScript button when it is hidden. As a result, this
will be hidden by default until a user installs another extension (or
shows the NoScript button and unpins it).
- - - - -
d206e967 by hackademix at 2025-12-10T15:27:14-03:00
BB 41834: Hide "Can't Be Removed - learn more" menu line for uninstallable add-ons
- - - - -
ea1c1d67 by Henry Wilkes at 2025-12-10T15:27:14-03:00
BB 41736: Customize toolbar for base-browser.
- - - - -
ea7f2133 by Henry Wilkes at 2025-12-10T15:27:15-03:00
BB 43864: Modify the urlbar for Base Browser.
- - - - -
584b546b by Henry Wilkes at 2025-12-10T15:27:16-03:00
BB 44040: Modify prompt service for Base Browser.
- - - - -
500e98df by Henry Wilkes at 2025-12-10T15:27:17-03:00
BB 43902: Modify the new sidebar for Base Browser.
- - - - -
bd911cdb by Beatriz Rizental at 2025-12-10T15:27:17-03:00
BB 44400: Disable liquid app glass icons for MacOS
This reverts commit 7ee252421988d7b3a38dc650986a3b4c3321a823.
- - - - -
2b09b040 by Pier Angelo Vendrame at 2025-12-10T15:27:18-03:00
BB 40925: Implemented the Security Level component
This component adds a new Security Level toolbar button which visually
indicates the current global security level via icon (as defined by the
extensions.torbutton.security_slider pref), a drop-down hanger with a
short description of the current security level, and a new section in
the about:preferences#privacy page where users can change their current
security level. In addition, the hanger and the preferences page will
show a visual warning when the user has modified prefs associated with
the security level and provide a one-click 'Restore Defaults' button to
get the user back on recommended settings.
Bug 40125: Expose Security Level pref in GeckoView
- - - - -
abec818b by Pier Angelo Vendrame at 2025-12-10T15:27:19-03:00
fixup! BB 40925: Implemented the Security Level component
Linted.
- - - - -
088db1ef by Henry Wilkes at 2025-12-10T15:27:20-03:00
fixup! BB 40925: Implemented the Security Level component
TB 44365: Fix CSS linting errors and missing variables.
- - - - -
3b688f70 by Pier Angelo Vendrame at 2025-12-10T15:27:21-03:00
BB 40926: Implemented the New Identity feature
- - - - -
e2a7c780 by Pier Angelo Vendrame at 2025-12-10T15:27:21-03:00
BB 42027: Base Browser migration procedures.
This commit implmenents the the Base Browser's version of _migrateUI.
- - - - -
420 changed files:
- + .gitlab-ci.yml
- + .gitlab/ci/jobs/helpers.py
- + .gitlab/ci/jobs/lint/lint.yml
- + .gitlab/ci/jobs/test/python-test.yml
- + .gitlab/ci/jobs/update-translations.yml
- + .gitlab/ci/mixins.yml
- + .gitlab/issue_templates/000 Bug Report.md
- + .gitlab/issue_templates/010 Proposal.md
- + .gitlab/issue_templates/020 Web Compatibility.md
- + .gitlab/issue_templates/030 Test.md
- + .gitlab/issue_templates/031 Fingerprinting.md
- + .gitlab/issue_templates/040 Feature.md
- + .gitlab/issue_templates/050 Backport.md
- + .gitlab/issue_templates/060 Rebase - Alpha.md
- + .gitlab/issue_templates/061 Rebase - Stable.md
- + .gitlab/issue_templates/063 Rebase - Rapid.md
- + .gitlab/issue_templates/090 Emergency Security Issue.md
- + .gitlab/issue_templates/Default.md
- + .gitlab/merge_request_templates/Default.md
- .prettierignore
- browser/Makefile.in
- browser/actors/moz.build
- browser/app/Makefile.in
- browser/app/macbuild/Contents/Info.plist.in
- browser/app/macbuild/Contents/MacOS-files.in
- browser/app/moz.build
- browser/app/permissions
- + browser/app/profile/001-base-profile.js
- browser/app/profile/firefox.js
- browser/base/content/aboutDialog-appUpdater.js
- browser/base/content/aboutDialog.js
- browser/base/content/aboutDialog.xhtml
- browser/base/content/appmenu-viewcache.inc.xhtml
- browser/base/content/browser-addons.js
- browser/base/content/browser-context.js
- browser/base/content/browser-init.js
- browser/base/content/browser-main.js
- browser/base/content/browser-menubar.inc
- browser/base/content/browser-safebrowsing.js
- browser/base/content/browser-sets.inc
- browser/base/content/browser-sets.js
- browser/base/content/browser.js
- browser/base/content/browser.js.globals
- browser/base/content/browser.xhtml
- + browser/base/content/languageNotification.js
- browser/base/content/main-popupset.inc.xhtml
- browser/base/content/navigator-toolbox.inc.xhtml
- browser/base/content/nsContextMenu.sys.mjs
- browser/base/jar.mn
- browser/base/moz.build
- − browser/branding/aurora/Assets.car
- − browser/branding/nightly/Assets.car
- − browser/branding/official/Assets.car
- − browser/branding/unofficial/Assets.car
- browser/components/BrowserComponents.manifest
- browser/components/BrowserContentHandler.sys.mjs
- browser/components/BrowserGlue.sys.mjs
- browser/components/DesktopActorRegistry.sys.mjs
- browser/components/ProfileDataUpgrader.sys.mjs
- browser/components/about/AboutRedirector.cpp
- browser/components/about/components.conf
- browser/components/aboutlogins/AboutLoginsParent.sys.mjs
- browser/components/aboutlogins/content/aboutLogins.css
- browser/components/aboutlogins/content/aboutLogins.mjs
- browser/components/aboutlogins/content/components/fxaccounts-button.css
- browser/components/aboutlogins/content/components/login-command-button.mjs
- browser/components/aboutlogins/content/components/login-list.mjs
- browser/components/customizableui/CustomizableUI.sys.mjs
- browser/components/customizableui/CustomizeMode.sys.mjs
- browser/components/customizableui/content/panelUI.inc.xhtml
- browser/components/downloads/DownloadSpamProtection.sys.mjs
- browser/components/extensions/parent/ext-browserAction.js
- browser/components/migration/ChromeMigrationUtils.sys.mjs
- browser/components/migration/content/migration-wizard.mjs
- browser/components/moz.build
- + browser/components/newidentity/content/newidentity.js
- + browser/components/newidentity/jar.mn
- + browser/components/newidentity/moz.build
- browser/components/newtab/AboutHomeStartupCache.sys.mjs
- browser/components/places/content/historySidebar.js
- browser/components/places/content/historySidebar.xhtml
- browser/components/preferences/home.inc.xhtml
- browser/components/preferences/home.js
- browser/components/preferences/jar.mn
- + browser/components/preferences/letterboxing-middle-dark.svg
- + browser/components/preferences/letterboxing-middle-light.svg
- + browser/components/preferences/letterboxing-top-dark.svg
- + browser/components/preferences/letterboxing-top-light.svg
- + browser/components/preferences/letterboxing.css
- + browser/components/preferences/letterboxing.inc.xhtml
- + browser/components/preferences/letterboxing.js
- browser/components/preferences/main.inc.xhtml
- browser/components/preferences/main.js
- browser/components/preferences/preferences.js
- browser/components/preferences/preferences.xhtml
- browser/components/preferences/privacy.inc.xhtml
- browser/components/preferences/privacy.js
- browser/components/reportbrokensite/ReportBrokenSite.sys.mjs
- browser/components/search/SERPCategorization.sys.mjs
- browser/components/search/SearchSERPTelemetry.sys.mjs
- + browser/components/securitylevel/SecurityLevelUIUtils.sys.mjs
- + browser/components/securitylevel/content/securityLevel.js
- + browser/components/securitylevel/content/securityLevelButton.css
- + browser/components/securitylevel/content/securityLevelButton.inc.xhtml
- + browser/components/securitylevel/content/securityLevelDialog.js
- + browser/components/securitylevel/content/securityLevelDialog.xhtml
- + browser/components/securitylevel/content/securityLevelIcon.svg
- + browser/components/securitylevel/content/securityLevelPanel.css
- + browser/components/securitylevel/content/securityLevelPanel.inc.xhtml
- + browser/components/securitylevel/content/securityLevelPreferences.css
- + browser/components/securitylevel/content/securityLevelPreferences.inc.xhtml
- + browser/components/securitylevel/jar.mn
- + browser/components/securitylevel/moz.build
- browser/components/sidebar/browser-sidebar.js
- browser/components/sidebar/sidebar-main.mjs
- browser/components/tabbrowser/content/tabbrowser.js
- browser/components/tabbrowser/content/tabgroup-menu.js
- browser/components/tabbrowser/moz.build
- browser/components/translations/content/selectTranslationsPanel.js
- browser/components/urlbar/ActionsProviderContextualSearch.sys.mjs
- browser/components/urlbar/QuickActionsLoaderDefault.sys.mjs
- browser/components/urlbar/QuickSuggest.sys.mjs
- browser/components/urlbar/SearchModeSwitcher.sys.mjs
- browser/components/urlbar/UrlbarController.sys.mjs
- browser/components/urlbar/UrlbarProvidersManager.sys.mjs
- browser/components/urlbar/UrlbarUtils.sys.mjs
- browser/components/urlbar/moz.build
- + browser/config/mozconfigs/base-browser
- + browser/config/mozconfigs/base-browser-android
- browser/extensions/moz.build
- + browser/fonts/fonts.conf
- browser/fonts/moz.build
- browser/installer/Makefile.in
- browser/installer/package-manifest.in
- browser/installer/windows/nsis/shared.nsh
- − browser/locales/en-US/chrome/overrides/appstrings.properties
- browser/locales/jar.mn
- browser/modules/AboutNewTab.sys.mjs
- browser/modules/BrowserWindowTracker.sys.mjs
- + browser/modules/ClipboardPrivacy.sys.mjs
- browser/modules/HomePage.sys.mjs
- + browser/modules/SecurityLevelNotification.sys.mjs
- browser/modules/moz.build
- browser/moz.build
- browser/moz.configure
- + browser/themes/shared/icons/new_identity.svg
- browser/themes/shared/identity-block/identity-block.css
- browser/themes/shared/jar.inc.mn
- browser/themes/shared/preferences/preferences.css
- browser/themes/shared/tabbrowser/content-area.css
- browser/themes/shared/tabbrowser/tabs.css
- browser/themes/shared/toolbarbutton-icons.css
- build/application.ini.in
- build/moz.build
- + build/moz.configure/basebrowser-resources.configure
- build/moz.configure/bootstrap.configure
- build/moz.configure/init.configure
- build/moz.configure/rust.configure
- build/moz.configure/update-programs.configure
- build/moz.configure/windows.configure
- build/variables.py
- caps/nsScriptSecurityManager.cpp
- config/external/moz.build
- devtools/client/aboutdebugging/src/actions/runtimes.js
- docshell/base/URIFixup.sys.mjs
- docshell/base/nsAboutRedirector.cpp
- docshell/build/components.conf
- dom/base/nsContentUtils.cpp
- dom/base/nsContentUtils.h
- dom/base/nsCopySupport.cpp
- dom/canvas/ClientWebGLContext.cpp
- dom/media/systemservices/video_engine/desktop_device_info.cc
- dom/media/webrtc/libwebrtc_overrides/modules/desktop_capture/desktop_capture_types.h
- dom/media/webrtc/libwebrtcglue/VideoConduit.cpp
- dom/media/webrtc/sdp/RsdparsaSdpAttributeList.cpp
- dom/media/webrtc/transport/nrinterfaceprioritizer.cpp
- dom/media/webrtc/transport/sigslot.h
- dom/media/webrtc/transport/test/ice_unittest.cpp
- dom/media/webrtc/transport/third_party/nICEr/src/net/local_addr.c
- dom/media/webrtc/transport/third_party/nICEr/src/net/local_addr.h
- dom/media/webrtc/transport/third_party/nICEr/src/stun/addrs-netlink.c
- dom/media/webrtc/transport/third_party/nICEr/src/stun/addrs-win32.c
- dom/media/webrtc/transport/third_party/nICEr/src/stun/addrs.c
- dom/media/webrtc/transport/third_party/nrappkit/src/log/r_log.c
- dom/media/webrtc/transport/third_party/nrappkit/src/registry/registry.c
- dom/webidl/moz.build
- eslint-file-globals.config.mjs
- eslint-ignores.config.mjs
- eslint.config.mjs
- extensions/auth/nsHttpNegotiateAuth.cpp
- + gfx/thebes/StandardFonts-macos-bb.inc
- gfx/thebes/StandardFonts-macos.inc
- + gfx/thebes/StandardFonts-win10-bb.inc
- gfx/thebes/StandardFonts-win10.inc
- gfx/thebes/gfxDWriteFontList.cpp
- gfx/thebes/gfxFcPlatformFontList.cpp
- gfx/thebes/gfxMacPlatformFontList.mm
- gfx/thebes/gfxPlatformFontList.cpp
- intl/locale/LocaleService.cpp
- layout/base/nsLayoutUtils.cpp
- mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/SearchUseCases.kt
- mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/ext/SearchEngine.kt
- mobile/android/android-components/components/feature/search/src/main/java/mozilla/components/feature/search/internal/SearchUrlBuilder.kt
- + mobile/android/basebrowser.configure
- mobile/android/exoplayer2/src/main/java/org/mozilla/thirdparty/com/google/android/exoplayer2/upstream/DefaultHttpDataSource.java
- mobile/android/fenix/app/src/main/java/org/mozilla/fenix/components/UseCases.kt
- mobile/android/fenix/app/src/main/java/org/mozilla/fenix/gecko/GeckoProvider.kt
- mobile/android/geckoview/api.txt
- − mobile/android/geckoview/src/androidTest/java/org/mozilla/geckoview/test/VideoCaptureTest.kt
- mobile/android/geckoview/src/main/AndroidManifest.xml
- mobile/android/geckoview/src/main/java/org/mozilla/gecko/media/GeckoMediaDrmBridgeV21.java
- mobile/android/geckoview/src/main/java/org/mozilla/geckoview/GeckoRuntimeSettings.java
- mobile/android/geckoview/src/main/java/org/mozilla/geckoview/GeckoSession.java
- mobile/android/geckoview/src/main/java/org/mozilla/geckoview/WebRequest.java
- mobile/android/gradle/with_gecko_binaries.gradle
- mobile/android/installer/package-manifest.in
- mobile/android/mach_commands.py
- mobile/android/moz.configure
- mobile/shared/chrome/geckoview/geckoview.js
- mobile/shared/components/geckoview/GeckoViewStartup.sys.mjs
- modules/libpref/init/StaticPrefList.yaml
- modules/libpref/init/all.js
- moz.configure
- + mozconfig-android-aarch64
- + mozconfig-android-all
- + mozconfig-android-armv7
- + mozconfig-android-x86
- + mozconfig-android-x86_64
- + mozconfig-linux-aarch64
- + mozconfig-linux-aarch64-dev
- + mozconfig-linux-arm
- + mozconfig-linux-i686
- + mozconfig-linux-x86_64
- + mozconfig-linux-x86_64-asan
- + mozconfig-linux-x86_64-dev
- + mozconfig-macos
- + mozconfig-macos-dev
- + mozconfig-windows-i686
- + mozconfig-windows-x86_64
- netwerk/base/nsIPrompt.idl
- netwerk/protocol/http/nsHttpHandler.cpp
- netwerk/protocol/http/nsHttpNTLMAuth.cpp
- netwerk/sctp/src/moz.build
- netwerk/sctp/src/netinet/sctp_cc_functions.c
- netwerk/sctp/src/netinet/sctp_indata.c
- netwerk/sctp/src/netinet/sctp_input.c
- netwerk/sctp/src/netinet/sctp_output.c
- netwerk/sctp/src/netinet/sctp_usrreq.c
- netwerk/sctp/src/netinet/sctputil.c
- netwerk/sctp/src/user_mbuf.c
- netwerk/sctp/src/user_socket.c
- netwerk/system/android/nsAndroidNetworkLinkService.cpp
- netwerk/system/linux/nsNetworkLinkService.cpp
- netwerk/system/mac/nsNetworkLinkService.mm
- netwerk/system/netlink/NetlinkService.cpp
- netwerk/system/win32/nsNotifyAddrListener.cpp
- netwerk/url-classifier/components.conf
- python/mach/mach/sentry.py
- python/mach/mach/telemetry.py
- python/mozboot/mozboot/bootstrap.py
- python/mozbuild/mozbuild/action/tooltool.py
- python/mozbuild/mozbuild/artifact_commands.py
- python/mozbuild/mozbuild/artifacts.py
- python/mozbuild/mozbuild/backend/base.py
- python/mozbuild/mozbuild/mach_commands.py
- + python/mozbuild/mozbuild/tbbutils.py
- python/mozbuild/mozbuild/test/python.toml
- + python/mozbuild/mozbuild/test/test_tbbutils.py
- python/mozlint/mozlint/cli.py
- python/sites/mach.txt
- security/manager/ssl/RemoteSecuritySettings.sys.mjs
- security/moz.build
- security/nss/lib/certhigh/ocsp.c
- security/nss/lib/libpkix/pkix_pl_nss/module/pkix_pl_socket.c
- security/nss/lib/ssl/Makefile
- services/settings/Attachments.sys.mjs
- services/settings/IDBHelpers.sys.mjs
- services/settings/RemoteSettingsClient.sys.mjs
- services/settings/dumps/gen_last_modified.py
- services/settings/dumps/main/moz.build
- services/settings/remote-settings.sys.mjs
- services/settings/servicesSettings.manifest
- services/settings/static-dumps/main/moz.build
- + services/settings/static-dumps/main/query-stripping.json
- services/sync/components.conf
- testing/specialpowers/api.js
- testing/testsuite-targets.mk
- testing/web-platform/tests/tools/third_party/attrs/.git_archival.txt
- third_party/abseil-cpp/absl/base/internal/thread_identity.cc
- third_party/abseil-cpp/absl/base/internal/thread_identity.h
- third_party/libwebrtc/modules/audio_coding/codecs/isac/main/source/os_specific_inline.h
- third_party/libwebrtc/modules/desktop_capture/win/desktop_capture_utils.cc
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capture_session.cc
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capturer_win.cc
- third_party/libwebrtc/modules/desktop_capture/win/wgc_capturer_win.h
- third_party/libwebrtc/modules/video_capture/windows/device_info_ds.h
- third_party/libwebrtc/rtc_base/platform_thread_types.cc
- third_party/libwebrtc/rtc_base/socket.h
- third_party/libwebrtc/rtc_base/system/file_wrapper.cc
- third_party/libwebrtc/rtc_base/win/create_direct3d_device.h
- third_party/sipcc/cpr_win_types.h
- third_party/sipcc/sdp_token.c
- toolkit/.eslintrc.mjs
- + toolkit/actors/FilesFilterChild.sys.mjs
- + toolkit/actors/FilesFilterParent.sys.mjs
- toolkit/actors/moz.build
- toolkit/components/antitracking/antitracking.manifest
- toolkit/components/antitracking/components.conf
- toolkit/components/downloads/DownloadCore.sys.mjs
- toolkit/components/downloads/DownloadIntegration.sys.mjs
- toolkit/components/extensions/Extension.sys.mjs
- toolkit/components/extensions/ExtensionParent.sys.mjs
- toolkit/components/extensions/ext-toolkit.json
- toolkit/components/extensions/parent/ext-trial-ml.js
- toolkit/components/moz.build
- toolkit/components/pdfjs/content/PdfjsParent.sys.mjs
- toolkit/components/pdfjs/content/web/viewer.mjs
- toolkit/components/places/moz.build
- toolkit/components/prompts/content/commonDialog.js
- toolkit/components/prompts/src/Prompter.sys.mjs
- toolkit/components/remotebrowserutils/RemoteWebNavigation.sys.mjs
- toolkit/components/resistfingerprinting/RFPHelper.sys.mjs
- toolkit/components/resistfingerprinting/content/letterboxing.css
- toolkit/components/search/ConfigSearchEngine.sys.mjs
- toolkit/components/search/SearchEngineSelector.sys.mjs
- toolkit/components/search/SearchService.sys.mjs
- + toolkit/components/search/content/base-browser-search-engine-icons.json
- + toolkit/components/search/content/base-browser-search-engines.json
- + toolkit/components/search/content/duckduckgo.ico
- + toolkit/components/search/content/startpage-16.png
- + toolkit/components/search/content/startpage-32.png
- + toolkit/components/search/jar.mn
- toolkit/components/search/moz.build
- + toolkit/components/search/tests/xpcshell/test_base_browser.js
- + toolkit/components/search/tests/xpcshell/test_security_level.js
- toolkit/components/search/tests/xpcshell/xpcshell.toml
- + toolkit/components/securitylevel/SecurityLevel.manifest
- + toolkit/components/securitylevel/SecurityLevel.sys.mjs
- + toolkit/components/securitylevel/components.conf
- + toolkit/components/securitylevel/moz.build
- toolkit/components/telemetry/app/TelemetrySend.sys.mjs
- toolkit/components/telemetry/moz.build
- toolkit/components/translations/actors/TranslationsChild.sys.mjs
- toolkit/components/translations/actors/TranslationsParent.sys.mjs
- toolkit/components/translations/actors/moz.build
- toolkit/components/translations/jar.mn
- toolkit/components/urlformatter/URLFormatter.sys.mjs
- toolkit/components/windowwatcher/nsIPromptService.idl
- toolkit/components/windowwatcher/nsWindowWatcher.cpp
- toolkit/content/aboutSupport.js
- toolkit/content/jar.mn
- toolkit/content/widgets/dialog.js
- toolkit/content/widgets/moz-message-bar/moz-message-bar.mjs
- toolkit/content/widgets/moz-support-link/moz-support-link.mjs
- toolkit/content/widgets/notificationbox.js
- + toolkit/locales-preview/base-browser-no-translate.ftl
- + toolkit/locales/en-US/toolkit/global/base-browser.ftl
- toolkit/locales/jar.mn
- toolkit/modules/ActorManagerParent.sys.mjs
- toolkit/modules/AppConstants.sys.mjs
- toolkit/modules/UpdateUtils.sys.mjs
- toolkit/modules/moz.build
- toolkit/mozapps/extensions/AddonManager.sys.mjs
- toolkit/mozapps/extensions/content/aboutaddons.html
- toolkit/mozapps/extensions/content/aboutaddons.js
- toolkit/mozapps/extensions/content/components/addon-mlmodel-details.mjs
- toolkit/mozapps/extensions/content/components/mlmodel-card-list-additions.mjs
- toolkit/mozapps/extensions/extensions.manifest
- toolkit/mozapps/extensions/internal/XPIDatabase.sys.mjs
- toolkit/mozapps/extensions/internal/XPIProvider.sys.mjs
- toolkit/mozapps/extensions/internal/moz.build
- toolkit/mozapps/extensions/test/browser/head.js
- toolkit/mozapps/extensions/test/xpcshell/head_addons.js
- toolkit/mozapps/installer/packager.mk
- toolkit/mozapps/installer/upload-files.mk
- toolkit/mozapps/update/UpdateService.sys.mjs
- toolkit/mozapps/update/UpdateServiceStub.sys.mjs
- toolkit/mozapps/update/common/updatehelper.cpp
- toolkit/mozapps/update/updater/launchchild_osx.mm
- toolkit/mozapps/update/updater/moz.build
- toolkit/mozapps/update/updater/nightly_aurora_level3_primary.der
- toolkit/mozapps/update/updater/nightly_aurora_level3_secondary.der
- toolkit/mozapps/update/updater/updater.cpp
- toolkit/xre/LauncherRegistryInfo.cpp
- toolkit/xre/MacLaunchHelper.h
- toolkit/xre/MacLaunchHelper.mm
- toolkit/xre/MacRunFromDmgUtils.mm
- toolkit/xre/ProfileReset.cpp
- toolkit/xre/dllservices/mozglue/WindowsDllBlocklist.cpp
- toolkit/xre/nsAppRunner.cpp
- toolkit/xre/nsIXREDirProvider.idl
- toolkit/xre/nsUpdateDriver.cpp
- toolkit/xre/nsXREDirProvider.cpp
- toolkit/xre/nsXREDirProvider.h
- + tools/base_browser/git-rebase-fixup-preprocessor
- + tools/base_browser/l10n/combine-translation-versions.py
- + tools/base_browser/l10n/combine/__init__.py
- + tools/base_browser/l10n/combine/combine.py
- + tools/base_browser/l10n/combine/tests/__init__.py
- + tools/base_browser/l10n/combine/tests/python.toml
- + tools/base_browser/l10n/combine/tests/test_android.py
- + tools/base_browser/l10n/combine/tests/test_dtd.py
- + tools/base_browser/l10n/combine/tests/test_fluent.py
- + tools/base_browser/l10n/combine/tests/test_properties.py
- + tools/base_browser/missing-css-variables.py
- + tools/base_browser/tb-dev
- tools/lint/fluent-lint/exclusions.yml
- tools/moz.build
- + tools/torbrowser/generate-bugzilla-triage-csv.sh
- uriloader/base/nsURILoader.cpp
- uriloader/exthandler/nsExternalHelperAppService.cpp
- widget/ScrollbarDrawingWin.cpp
- widget/ScrollbarDrawingWin11.cpp
- widget/android/WebExecutorSupport.cpp
- widget/gtk/MPRISServiceHandler.cpp
- widget/gtk/nsLookAndFeel.cpp
- widget/windows/WinTaskbar.cpp
- widget/windows/nsDataObj.cpp
- xpcom/io/nsAppFileLocationProvider.cpp
- xpfe/appshell/AppWindow.cpp
The diff was not included because it is too large.
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/f1…
--
View it on GitLab: https://gitlab.torproject.org/tpo/applications/mullvad-browser/-/compare/f1…
You're receiving this email because of your account on gitlab.torproject.org.
1
0