- package: tis-firefox-multi
- name: Mozilla Firefox Multi-Language
- version: 142.0-23
- categories: Internet
- maintainer: WAPT Team, Tranquil IT, Jimmy PELÉ, Amel FRADJ
- editor: Mozilla Foundation,Mozilla Corporation
- licence: MPL 2.0
- locale: all
- target_os: windows
- impacted_process: firefox
- architecture: x86
- signature_date:
- size: 80.26 Mo
- installed_size: 220.64 Mo
- homepage : https://www.mozilla.org/
- conflicts :
package : tis-firefox-multi
version : 142.0-23
architecture : x86
section : base
priority : optional
name : Mozilla Firefox Multi-Language
categories : Internet
maintainer : WAPT Team, Tranquil IT, Jimmy PELÉ, Amel FRADJ
description : Mozilla Firefox, is a free and open-source web browser - The package is multi-language and pre-configured
depends :
conflicts : tis-firefox, tis-firefox-esr, tis-firefox-esr-multi, tis-config-for-firefox, tis-config-firefox
maturity : PROD
locale : all
target_os : windows
min_wapt_version : 2.3
sources : https://www.mozilla.org/firefox/all/#product-desktop-release
installed_size : 220643328
impacted_process : firefox
description_fr : Mozilla Firefox, est un navigateur web gratuit et open source - Le paquet est multilingue et préconfiguré
description_pl : Mozilla Firefox, jest darmową i otwartą przeglądarką internetową - Pakiet jest wielojęzyczny i wstępnie skonfigurowany
description_de : Mozilla Firefox, ist ein kostenloser und quelloffener Webbrowser - Das Paket ist mehrsprachig und vorkonfiguriert
description_es : Mozilla Firefox, es un navegador web gratuito y de código abierto - El paquete es multilingüe y está preconfigurado
description_pt : Mozilla Firefox, é um navegador web gratuito e de código aberto - O pacote é multilingue e pré-configurado
description_it : Mozilla Firefox è un browser web gratuito e open-source - Il pacchetto è multilingue e preconfigurato
description_nl : Mozilla Firefox, is een gratis en open-source webbrowser - Het pakket is meertalig en voorgeconfigureerd
description_ru : Mozilla Firefox - это бесплатный веб-браузер с открытым исходным кодом - Пакет является многоязычным и предварительно настроенным
audit_schedule :
editor : Mozilla Foundation,Mozilla Corporation
keywords : web,browser,navigateur,firefox,mozilla
licence : MPL 2.0
homepage : https://www.mozilla.org/
package_uuid : d28d4778-60f0-41c2-a06d-6a561ffd2b14
valid_from :
valid_until :
forced_install_on :
changelog : https://www.mozilla.org/firefox/releases/
min_os_version : 10.0
max_os_version :
icon_sha256sum : 2dc82eade364831757e067169d0039e6e0c9b693f343ea9a62db709eb1942c9e
signer : Tranquil IT
signer_fingerprint: 8c5127a75392be9cc9afd0dbae1222a673072c308c14d88ab246e23832e8c6bb
signature_date : 2025-08-19T16:00:20.000000
signed_attributes : package,version,architecture,section,priority,name,categories,maintainer,description,depends,conflicts,maturity,locale,target_os,min_wapt_version,sources,installed_size,impacted_process,description_fr,description_pl,description_de,description_es,description_pt,description_it,description_nl,description_ru,audit_schedule,editor,keywords,licence,homepage,package_uuid,valid_from,valid_until,forced_install_on,changelog,min_os_version,max_os_version,icon_sha256sum,signer,signer_fingerprint,signature_date,signed_attributes
signature : qyQSgkrTDjlGrZETG/zqn09E0KrtBK9G1Ec0O2bvS9AbDfqUXn9Diepg2CIvKAE4gd3dSkMsc33R/3z9ckgbt/h4WFBOSZ5TKZ0CnKaJdB3TLfhWbVDg38DWG2PcDZrpiiQiaMp0OGAr90BABenmHZjNxVD5nHNDeRDToA3pTLboE1Jy8stPPPovTGMwV1KrZXRTM12G0CBLIhGpGM77z/mQVYsl4dD2CSrwwUBINLYT+llPh15z44ZLvAafQNlLKVkifg1JXlUOQgnSOeEmLfTdiwbJ4IMbig5Qu3HiJVHU1MMnJfQRinL6u60zdKGYfd8APo8PkcJRc5O95h/SfQ==
# -*- coding: UTF-8 -*-
from setuphelpers import *
import time
# Installation procedure: https://firefox-source-docs.mozilla.org/browser/installer/windows/installer/FullConfig.html
# Configuration procedure: https://support.mozilla.org/kb/customizing-firefox-using-policiesjson
dist_dir = "distribution"
dist_path = makepath(programfiles, "Mozilla Firefox", dist_dir)
ext_dir = "extensions"
ext_path = makepath(dist_path, ext_dir)
policies_file = "policies.json"
policies_path = makepath(dist_path, policies_file)
list_pre_installed_lang = ["fr", "en-GB", "es-ES", "de", "it"] # Reminder: 'en-US' will be native
def install():
bin_name = glob.glob("Firefox_Setup_*.exe")[0]
app_uninstallkey = "Mozilla Firefox %s (%s en-US)" % (control.get_software_version(), control.architecture)
# Uninstalling others versions
for uninstall in installed_softwares(name="Mozilla Firefox"):
if not control.architecture in uninstall["name"] or "ESR" in uninstall["name"]:
print("Removing: %s" % (uninstall["name"]))
killalltasks(control.get_impacted_process_list())
run(uninstall_cmd(uninstall["key"]))
time.sleep(15) # Required for Firefox
# Installing the software
install_exe_if_needed(
bin_name,
silentflags="/S /PreventRebootRequired=false /TaskbarShortcut=false /DesktopShortcut=false /StartMenuShortcut=true /MaintenanceService=false /RegisterDefaultAgent=false /RemoveDistributionDir=false /OptionalExtensions=false",
key=app_uninstallkey,
min_version=control.get_software_version(),
)
# Making sure that Mozilla Maintenance Service is not installed
for uninstall in installed_softwares(name="Mozilla Maintenance Service"):
print("Removing: %s" % (uninstall["name"]))
run(uninstall_cmd(uninstall["key"]))
# Removing policies from registry since it bypass the policies.json file (source: https://support.mozilla.org/bm/questions/1236197)
ffox_policies_reg_path = r"SOFTWARE\Policies\Mozilla\Firefox"
if reg_key_exists(HKEY_LOCAL_MACHINE, ffox_policies_reg_path):
registry_deletekey(HKEY_LOCAL_MACHINE, ffox_policies_reg_path, "", force=True, recursive=True)
# Adding distribution folder for language packs, dictionaries, extensions and configuration
if isdir(dist_path):
remove_tree(dist_path)
mkdirs(dist_path)
filecopyto(makepath(basedir, policies_file), dist_path)
copytree2(makepath(basedir, ext_dir), ext_path)
# Changing default language
data = json_load_file(policies_path)
my_lang = get_language()
for select_lang in list_pre_installed_lang:
if my_lang in select_lang:
if select_lang == "en-GB":
select_lang = "en-US"
default_lang = {"RequestedLocales": [select_lang]}
data["policies"].update(default_lang)
json_write_file(policies_path, data, indent=2)
# -*- coding: UTF-8 -*-
from setuphelpers import *
from setupdevhelpers import *
import requests
import waptlicences
dist_dir = "distribution"
dist_path = makepath(programfiles, "Mozilla Firefox", dist_dir)
ext_dir = "extensions"
ext_path = makepath(dist_path, ext_dir)
policies_file = "policies.json"
policies_path = makepath(dist_path, policies_file)
list_pre_installed_lang = ["fr", "en-GB", "es-ES", "de", "it"] # Reminder: 'en-US' will be native
def update_package():
# Declaring local variables
package_updated = False
proxies = get_proxies()
if not proxies:
proxies = get_proxies_from_wapt_console()
app_name = control.name
arch_dict = {"x64": "win64", "x86": "win", "all": "win"}
policies_content = json_load_file(policies_file)
# Getting the latest version from official sources
download_url = requests.head(f"https://download.mozilla.org/?product=firefox-latest-ssl&os={arch_dict[control.architecture]}&lang=en-US", proxies=proxies).headers["Location"]
latest_bin = download_url.rsplit("/", 1)[-1].replace("%20", "_")
latest_bin_extension = latest_bin.rsplit('.', 1)[-1]
version = download_url.split("/")[-4]
# Downloading the latest binaries
print("Latest %s version is: %s" % (app_name, version))
print("Download URL is: %s" % download_url)
if not isfile(latest_bin):
print("Downloading: %s" % latest_bin)
wget(download_url, latest_bin, proxies=proxies)
else:
print("Binary is present: %s" % latest_bin)
expected_issuer = "Mozilla Corporation"
sign_name = waptlicences.check_exe_signing_certificate(latest_bin)[0]
if sign_name != expected_issuer:
error(f'Bad issuer {sign_name} != {expected_issuer} ')
# Changing the version of the package
if Version(version) > Version(control.get_software_version()):
print("Software version updated (from: %s to: %s)" % (control.get_software_version(), Version(version)))
package_updated = True
else:
print("Software version up-to-date (%s)" % Version(version))
control.set_software_version(version)
control.save_control_to_wapt()
# Deleting outdated binaries
for f in glob.glob(f'*.{latest_bin_extension}'):
if f != latest_bin:
remove_file(f)
# Remove existing extensions folder
if isdir(ext_dir):
remove_tree(ext_dir)
download_firefox_language_pack(policies_content, proxies)
download_firefox_dictionnary(policies_content, proxies)
# Downloading ublock origin extensions
wget("https://addons.mozilla.org/firefox/downloads/latest/ublock-origin/latest.xpi", makepath(ext_dir, "ublock-origin.xpi"), proxies=proxies)
print(f"Creating {policies_file} file for language packs, dictionaries, extensions, and configuration")
if isfile(policies_file):
remove_file(policies_file)
json_write_file(policies_file, policies_content, indent=4)
# Validate or not the update-package-sources
return package_updated
def download_firefox_language_pack(policies_content, proxies):
print("\n### DOWNLOADING LANGUAGE PACK ###\n")
url_api_lp = "https://services.addons.mozilla.org/api/v3/addons/language-tools/?app=firefox&type=language"
print("API used for language packs is: " + url_api_lp)
version_major = control.version.split("-")[0].split(".")[0]
# Retrieved the download url of the language pack
json_read = wgets(url_api_lp, proxies=proxies, as_json=True)
for lang in list_pre_installed_lang:
# Retrieve the download URL for the language pack
url_lp = next((lp["url"] + "versions/" for lp in json_read["results"] if lp["target_locale"] == lang), None)
if not url_lp:
print(f"No URL found for language pack: {lang}")
continue
# Find the version-specific download link
url_dl_lp = None
for download_lp in bs_find_all(url_lp, "a", "class", "InstallButtonWrapper-download-link", proxies=proxies):
if version_major in download_lp.get("href", ""):
url_dl_lp = download_lp["href"]
break
# Check if a valid URL for the language pack was found
if not url_dl_lp:
print(f"No valid download link found for language pack: {lang}")
continue
lp_name = f"{lang}_language_pack.xpi"
print(f"Download URL for language pack in {lang} is: {url_dl_lp}")
wget(url_dl_lp, makepath(ext_dir, lp_name), proxies=proxies)
policies_content["policies"]["ExtensionSettings"].update({
f"langpack-{lang}@firefox.mozilla.org": {
"install_url": f"file://{ext_path}\\{lang}_language_pack.xpi",
"installation_mode": "force_installed"
}
})
def download_firefox_dictionnary(policies_content, proxies):
print("\n### DOWNLOADING DICTIONNARY ###\n")
url_api_dict_lang = "https://services.addons.mozilla.org/api/v3/addons/language-tools/?app=firefox&type=dictionary"
print("API used for language dictionaries is: " + url_api_dict_lang)
json_read = wgets(url_api_dict_lang, proxies=proxies, as_json=True)
for lang in list_pre_installed_lang:
if lang == "de":
lang = "de-DE"
dict_lang = next((d for d in json_read["results"] if d["target_locale"] == lang), None)
if not dict_lang:
print(f"No dictionary found for language: {lang}")
continue
url_dl_dict_lang = f"https://addons.mozilla.org/firefox/downloads/latest/{dict_lang['slug']}/latest.xpi"
ext_add_on_id = dict_lang["guid"]
dict_lang_name = f"{lang}_dictionary.xpi"
print(f"Download URL for dictionary in {lang} is: {url_dl_dict_lang}")
wget(url_dl_dict_lang, makepath(ext_dir, dict_lang_name), proxies=proxies)
policies_content["policies"]["ExtensionSettings"].update({
ext_add_on_id: {
"install_url": f"file://{ext_path}\\{dict_lang_name}",
"installation_mode": "force_installed"
}
})
6d2fcef5b1a8ac2498258ec880d9ed93b172ea2a760e798ae833946ae00330e9 : Firefox_Setup_142.0.exe
38d056ab130f7bf7c481c12636a4e9959de36561d3dfcbe54c6e3571bc0c1dc3 : WAPT/certificate.crt
5994df330789dc977c9e2b7d4ce5d8fa32ad87be532cd2330ed602712562a11e : WAPT/control
2dc82eade364831757e067169d0039e6e0c9b693f343ea9a62db709eb1942c9e : WAPT/icon.png
00ef6eb3c10171a87fb22ab6e516846678b73c56ae828cc19d11e32e43b8457a : extensions/de-DE_dictionary.xpi
22d1fc6393e9a2ad19e879595449b4e610b6b6b1dad7f4202ba6a9b1c1351cad : extensions/de_language_pack.xpi
9fb12b246d84eb1df6d67321fd3d7ca33524ecfd822e5e5d361f91bd9181cc37 : extensions/en-GB_dictionary.xpi
455b714cd3cce006f6a954fe014aa1d3c37c05788750d020c20db3eed60cd2f7 : extensions/en-GB_language_pack.xpi
83a8f9117d7224728507d90990a7dfecd84a05aa55f9dd8ccc1bc199a163e739 : extensions/es-ES_dictionary.xpi
8c8a0e73351942dea2d9a0f3833282e8c9a99ec412eebffdf07150e5f16ceff2 : extensions/es-ES_language_pack.xpi
e0e90b88b177dc1b268b019c8672eb1be943f12e174ad474dbdc46f0e6fbaa6f : extensions/fr_dictionary.xpi
457645cd433e546822c6ec37eb99ef2283973c8bf9beef1111742ab50f11c1b8 : extensions/fr_language_pack.xpi
90b173ffdde34a77108152a5ff51879767b1dd84e0aa0dfb7b2bab94cd2e7f53 : extensions/it_dictionary.xpi
53e72a8294557097eed5d1e93bb7905d78c6f42b24613d683aad1c448a966bda : extensions/it_language_pack.xpi
3e73c96a29a933866065f0756fe032984bf5b254af8dd1afd7a7f7e0668a33cf : extensions/ublock-origin.xpi
9be4d5b3cbf2891d6e499ac984ce2e6c6f2536ba416a28ad1c97896f9f7ea7c5 : luti.json
235c7795d36cf4c854ed0e8a4eadde6315457c6c9cf47238391757330be09d9e : policies.json
adc9b2bfc251776e43458be4184bcb1e28f8c684678fabafc1020d3dd27e2c07 : setup.py
e1c748605b036d72ccf16675ea84879b8d461f77ce8d7108c3cf2958f40113f5 : update_package.py