2023-01-14 02:14:43 -08:00
{{- if ne .host.distro.family "windows" -}}
#!/usr/bin/env bash
2023-03-27 02:56:38 -07:00
# @file run_onchange_after_40-firefox.tmpl
2023-03-27 03:05:48 -07:00
# @brief This script configures system-wide settings, sets up Firefox Profile Switcher, creates various profiles from different sources, and installs a configurable list of Firefox Add-Ons.
2023-03-27 02:56:38 -07:00
# @description
# The Firefox setup script performs a handful of tasks that automate the setup of Firefox as well as
# useful utilities that will benefit Firefox power-users. The script also performs the same logic on
# [LibreWolf](https://librewolf.net/) installations. The features that are included are:
#
# * Installs and sets up [Firefox Profile Switcher](https://github.com/null-dev/firefox-profile-switcher)
# * Sets up system-wide enterprise settings (with configurations found in `~/.local/share/firefox`)
# * Sets up a handful of default profiles to use with the Firefox Profile Switcher
# * Automatically installs the plugins defined in the firefoxAddOns key of [`home/.chezmoidata.yaml`](https://github.com/megabyte-labs/install.doctor/blob/master/home/.chezmoidata.yaml) to the Standard and Private profiles
# * Configures the default profile to clone its settings from the profile stored in firefoxPublicProfile of `home/.chezmoidata.yaml`
# * Optionally, if the Chezmoi encryption key is present, then the default profile will be set to the contents of an encrypted `.tar.gz` that must be stored in the cloud somewhere (with the firefoxPrivateProfile key in `home/.chezmoidata.yaml` defining the URL of the encrypted `.tar.gz`)
#
# ## Profiles
#
# The script sets up numerous profiles for user flexibility. They can be switched by using the Firefox Profile Switcher
# that this script sets up. The map of the profiles is generated by using the template file stored in `~/.local/share/firefox/profiles.ini`.
# The following details the features of each profile:
#
# | Name | Description |
# |------------------|---------------------------------------------------------------------------------------------|
# | Factory | Default browser settings (system-wide configurations still apply) |
# | default-release | Same as Factory (unmodified and generated by headlessly opening Firefox / LibreWolf) |
# | Git (Public) | Pre-configured profile with address stored in `firefoxPublicProfile` |
# | Standard | Cloned from the profile above with `firefoxAddOns` also installed |
# | Miscellaneous | Cloned from the Factory profile (with the user.js found in `~/.config/firefox` applied) |
# | Development | Same as Miscellaneous |
# | Automation | Same as Miscellaneous |
# | Private | Populated from an encrypted profile stored in the cloud (also installs `firefoxAddOns`) |
#
# ## Notes
#
# * The Firefox Profile Switcher is only compatible with Firefox and not LibreWolf
# * This script is only designed to properly provision profiles on a fresh installation (so it does not mess around with pre-existing / already configured profiles)
#
# @see [Script on GitHub](https://github.com/megabyte-labs/install.doctor/blob/master/home/.chezmoiscripts/universal/run_onchange_after_40-firefox.tmpl)
# @see [System-wide configurations](https://github.com/megabyte-labs/install.doctor/tree/master/home/dot_local/share/firefox) as well as the location of the `profile.ini` and some other configurations
# @see [User-specific configurations](https://github.com/megabyte-labs/install.doctor/blob/master/home/dot_config/firefox/user.js) added to all profiles except Factory
2023-01-14 02:14:43 -08:00
{{ includeTemplate "universal/profile" }}
{{ includeTemplate "universal/logg" }}
2023-01-29 21:21:11 -08:00
# Firefox plugins: {{ list (.firefoxAddOns | toString | replace "[" "" | replace "]" "") | uniq | join " " }}
2023-02-01 00:48:57 -08:00
### Installs the Firefox Profile Connector on Linux systems (Snap / Flatpak installs are not included in this function, but instead inline below)
function installFirefoxProfileConnector() {
logg info 'Installing the Firefox Profile Connector'
if command -v apt-get > /dev/null; then
sudo apt-get install -y https://github.com/null-dev/firefox-profile-switcher-connector/releases/latest/download/linux-x64.deb
elif command -v dnf > /dev/null; then
sudo dnf install -y https://github.com/null-dev/firefox-profile-switcher-connector/releases/latest/download/linux-x64.rpm
elif command -v yay > /dev/null; then
yay -Ss firefox-profile-switcher-connector
else
logg warn 'apt-get, dnf, and yay were all unavailable so the Firefox Profile Connector helper executable could not be installed'
fi
}
2023-01-14 02:14:43 -08:00
### Add Firefox enterprise profile
2023-01-29 23:27:31 -08:00
# TODO - figure out how to do this for other installations like Flatpak and macOS and Librewolf
2023-03-27 02:56:38 -07:00
for FIREFOX_DIR in '/usr/lib/firefox' '/usr/lib/firefox-esr' '/etc/firefox' '/etc/firefox-esr' '/Applications/Firefox.app/Contents/Resources' '/Applications/LibreWolf.app/Contents/Resources/'; do
2023-01-30 21:44:52 -08:00
if [ -d " $ FIREFOX_DIR " ] && [ -d " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox" ] && command -v rsync > /dev/null; then
sudo rsync -artvu " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/" " $ FIREFOX_DIR "
2023-01-14 02:14:43 -08:00
fi
done
2023-01-25 00:46:31 -08:00
### Loop through various Firefox profile locations
for SETTINGS_DIR in " $ HOME /snap/firefox/common/.mozilla/firefox" " $ HOME /.var/app/org.mozilla.firefox/.mozilla/firefox" " $ HOME /.var/app/io.gitlab.librewolf-community/.librewolf" " $ HOME /Library/Application Support/Firefox/Profiles" " $ HOME /Library/Application Support/LibreWolf/Profiles" " $ HOME /.mozilla/firefox"; do
### Determine executable to use
2023-01-29 23:27:31 -08:00
logg info "Processing Firefox profile location $ SETTINGS_DIR "
unset FIREFOX_EXE
2023-01-25 00:46:31 -08:00
if [ " $ SETTINGS_DIR " == " $ HOME /.var/app/org.mozilla.firefox" ]; then
2023-01-29 21:31:10 -08:00
if ! command -v org.mozilla.firefox > /dev/null; then
continue
2023-01-29 23:27:31 -08:00
else
FIREFOX_EXE="$(which org.mozilla.firefox)"
2023-02-01 00:48:57 -08:00
### Firefox Profile Switcher
BASE_DIR=" $ HOME /.var/app/org.mozilla.firefox"
BIN_INSTALL_DIR=" $ BASE_DIR /data/firefoxprofileswitcher-install"
MANIFEST_INSTALL_DIR=" $ BASE_DIR /.mozilla/native-messaging-hosts"
DOWNLOAD_URL="https://github.com/null-dev/firefox-profile-switcher-connector/releases/latest/download/linux-x64.deb"
### Ensure Firefox Profile Switcher is not already installed
if [ ! -f " $ BIN_INSTALL_DIR /usr/bin/ff-pswitch-connector" ] || [ ! -f " $ MANIFEST_INSTALL_DIR /ax.nd.profile_switcher_ff.json" ]; then
### Download profile switcher
mkdir -p " $ BIN_INSTALL_DIR "
TMP_FILE="$(mktemp)"
logg info 'Downloading Firefox Profile Switch connector'
curl -sSL " $ DOWNLOAD_URL " -o " $ TMP_FILE "
ar p " $ TMP_FILE " data.tar.xz | tar xfJ - --strip-components=2 -C " $ BIN_INSTALL_DIR " usr/bin/ff-pswitch-connector
rm -f " $ TMP_FILE "
### Create manifest
logg info 'Copying profile switcher configuration to manifest directory'
mkdir -p " $ MANIFEST_INSTALL_DIR "
cat " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/profile-switcher.json" | sed 's/PATH_PLACEHOLDER/'" $ BIN_INSTALL_DIR "'/' > " $ MANIFEST_INSTALL_DIR /ax.nd.profile_switcher_ff.json"
fi
2023-01-29 21:31:10 -08:00
fi
2023-01-25 00:46:31 -08:00
elif [ " $ SETTINGS_DIR " == " $ HOME /.var/app/io.gitlab.librewolf-community/.librewolf" ]; then
2023-01-29 21:31:10 -08:00
if ! command -v io.gitlab.librewolf-community > /dev/null; then
continue
2023-01-29 23:27:31 -08:00
else
FIREFOX_EXE="$(which io.gitlab.librewolf-community)"
2023-01-29 21:31:10 -08:00
fi
2023-01-25 00:46:31 -08:00
elif [ " $ SETTINGS_DIR " == " $ HOME /Library/Application Support/Firefox/Profiles" ]; then
FIREFOX_EXE="/Applications/Firefox.app/Contents/MacOS/firefox"
2023-01-29 21:31:10 -08:00
if [ ! -f " $ FIREFOX_EXE " ]; then
continue
2023-02-01 00:48:57 -08:00
else
### Download Firefox Profile Switcher
if [ ! -d /usr/local/Cellar/firefox-profile-switcher-connector ]; then
logg info 'Ensuring Firefox Profile Switcher is installed'
brew install null-dev/firefox-profile-switcher/firefox-profile-switcher-connector
fi
### Ensure Firefox Profile Switcher configuration is symlinked
if [ ! -d "/Library/Application Support/Mozilla/NativeMessagingHosts/ax.nd.profile_switcher_ff.json" ]; then
logg info 'Ensuring Firefox Profile Switcher is configured'
sudo mkdir -p "/Library/Applcation Support/Mozilla/NativeMessagingHosts"
sudo ln -sf "$(brew ls -l firefox-profile-switcher-connector | grep -i ax.nd.profile_switcher_ff.json | head -n1)" "/Library/Application Support/Mozilla/NativeMessagingHosts/ax.nd.profile_switcher_ff.json"
fi
2023-02-01 12:24:51 -08:00
fi
2023-01-29 21:09:12 -08:00
elif [ " $ SETTINGS_DIR " == " $ HOME /Library/Application Support/LibreWolf/Profiles" ]; then
2023-01-25 00:46:31 -08:00
FIREFOX_EXE="/Applications/LibreWolf.app/Contents/MacOS/librewolf"
2023-01-29 21:31:10 -08:00
if [ ! -f " $ FIREFOX_EXE " ]; then
continue
fi
2023-01-25 00:46:31 -08:00
elif [ " $ SETTINGS_DIR " == " $ HOME /snap/firefox/common/.mozilla/firefox" ]; then
2023-01-29 23:27:31 -08:00
FIREFOX_EXE="/snap/bin/firefox"
if [ ! -f " $ FIREFOX_EXE " ]; then
2023-01-29 21:31:10 -08:00
continue
2023-02-01 00:48:57 -08:00
else
### Firefox Profile Switcher
BASE_DIR=" $ HOME /snap/firefox/common"
BIN_INSTALL_DIR=" $ BASE_DIR /firefoxprofileswitcher-install"
MANIFEST_INSTALL_DIR=" $ BASE_DIR /.mozilla/native-messaging-hosts"
DOWNLOAD_URL="https://github.com/null-dev/firefox-profile-switcher-connector/releases/latest/download/linux-x64.deb"
### Ensure Firefox Profile Switcher is not already installed
if [ ! -f " $ BIN_INSTALL_DIR /usr/bin/ff-pswitch-connector" ] || [ ! -f " $ MANIFEST_INSTALL_DIR /ax.nd.profile_switcher_ff.json" ]; then
### Download profile switcher
mkdir -p " $ BIN_INSTALL_DIR "
TMP_FILE="$(mktemp)"
logg info 'Downloading Firefox Profile Switch connector'
curl -sSL " $ DOWNLOAD_URL " -o " $ TMP_FILE "
ar p " $ TMP_FILE " data.tar.xz | tar xfJ - --strip-components=2 -C " $ BIN_INSTALL_DIR " usr/bin/ff-pswitch-connector
rm -f " $ TMP_FILE "
### Create manifest
logg info 'Copying profile switcher configuration to manifest directory'
mkdir -p " $ MANIFEST_INSTALL_DIR "
cat " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/profile-switcher.json" | sed 's/PATH_PLACEHOLDER/'" $ BIN_INSTALL_DIR "'/' > " $ MANIFEST_INSTALL_DIR /ax.nd.profile_switcher_ff.json"
fi
2023-01-29 21:31:10 -08:00
fi
elif [ " $ SETTINGS_DIR " == " $ HOME /.mozilla/firefox" ]; then
2023-01-25 00:46:31 -08:00
if command -v firefox-esr > /dev/null; then
FIREFOX_EXE="$(which firefox-esr)"
2023-02-01 00:48:57 -08:00
installFirefoxProfileConnector
2023-01-25 00:46:31 -08:00
elif command -v firefox > /dev/null && [ "$(which firefox)" != *'snap'* ] && [ "$(which firefox)" != *'flatpak'* ] && [ ! -d /Applications ] && [ ! -d /System ]; then
# Conditional check ensures Snap / Flatpak / macOS Firefox versions do not try to install to the wrong folder
FIREFOX_EXE="$(which firefox)"
2023-02-01 00:48:57 -08:00
installFirefoxProfileConnector
2023-01-25 00:46:31 -08:00
else
2023-03-27 02:56:38 -07:00
if [ -d /Applications ] && [ -d /System ]; then
# Continue on macOS without logging because profiles are not stored here on macOS
continue
else
logg warn 'Unable to register Firefox executable'
logg info "Settings directory: $ SETTINGS_DIR "
continue
fi
2023-01-25 00:46:31 -08:00
fi
2023-01-14 02:14:43 -08:00
fi
2023-01-25 00:46:31 -08:00
### Initiatize Firefox default profiles
if command -v " $ FIREFOX_EXE " > /dev/null; then
2023-02-01 05:43:29 -08:00
### Create default profile by launching Firefox headlessly
2023-01-29 23:27:31 -08:00
logg info "Firefox executable set to $ FIREFOX_EXE "
2023-01-25 00:46:31 -08:00
if [ ! -d " $ SETTINGS_DIR " ]; then
logg info 'Running Firefox headlessly to generate default profiles'
2023-01-29 23:27:31 -08:00
timeout 8 " $ FIREFOX_EXE " --headless
2023-01-25 00:46:31 -08:00
logg info 'Finished running Firefox headlessly'
fi
2023-02-01 05:43:29 -08:00
2023-03-27 02:56:38 -07:00
### Add the populated profiles.ini
logg info "Copying " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/profiles.ini" to profile directory"
if [ -d /Applications ] && [ -d /System ]; then
# macOS
logg info "Copying ~/.local/share/profiles.ini to $ SETTINGS_DIR /../profiles.ini"
cp -f " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/profiles.ini" " $ SETTINGS_DIR /../profiles.ini"
SETTINGS_INI=" $ SETTINGS_DIR /../installs.ini"
else
# Linux
logg info "Copying ~/.local/share/profiles.ini to $ SETTINGS_DIR /profiles.ini"
cp -f " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/profiles.ini" " $ SETTINGS_DIR /profiles.ini"
SETTINGS_INI=" $ SETTINGS_DIR /installs.ini"
fi
2023-02-01 05:43:29 -08:00
### Default profile (created by launching Firefox headlessly)
2023-03-27 02:56:38 -07:00
# DEFAULT_RELEASE_PROFILE="$(find " $ SETTINGS_DIR " -mindepth 1 -maxdepth 1 -name "*.default-*")"
DEFAULT_PROFILE_PROFILE=" $ SETTINGS_DIR /$(cat " $ SETTINGS_INI " | grep 'Default=' | sed 's/.*Profiles\///')"
logg info 'Removing previous installs.ini file'
rm -f " $ SETTINGS_INI "
2023-02-01 05:43:29 -08:00
# DEFAULT_PROFILE="$(find " $ SETTINGS_DIR " -mindepth 1 -maxdepth 1 -name "*.default" -not -name "profile.default")"
if [ -n " $ DEFAULT_RELEASE_PROFILE " ]; then
2023-03-27 02:56:38 -07:00
logg info "Syncing $ DEFAULT_RELEASE_PROFILE to $ SETTINGS_DIR /profile.default"
2023-02-01 05:43:29 -08:00
rsync -a " $ DEFAULT_RELEASE_PROFILE /" " $ SETTINGS_DIR /profile.default"
2023-01-29 23:27:31 -08:00
fi
2023-02-01 05:43:29 -08:00
### Miscellaneous default profiles
for NEW_PROFILE in "automation" "development" "miscellaneous"; do
if [ ! -d " $ SETTINGS_DIR /profile. ${ NEW_PROFILE } " ] && [ -d " $ SETTINGS_DIR /profile.default" ]; then
logg info "Cloning $ NEW_PROFILE from profile.default"
rsync -a " $ SETTINGS_DIR /profile.default/" " $ SETTINGS_DIR /profile. ${ NEW_PROFILE } "
2023-02-04 21:13:10 -08:00
rsync -a " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/" " $ SETTINGS_DIR /profile. ${ NEW_PROFILE } "
2023-03-27 02:56:38 -07:00
cp -f " ${ XDG_CONFIG_HOME : - $ HOME / . config } /firefox/user.js" " $ SETTINGS_DIR /profile. ${ NEW_PROFILE } "
2023-02-01 05:43:29 -08:00
fi
done
### Public git profile
if [ -d " $ SETTINGS_DIR /profile.git" ]; then
logg info 'Resetting the Firefox git profile'
cd " $ SETTINGS_DIR /profile.git"
git reset --hard HEAD
git clean -fxd
logg info 'Pulling latest updates to the Firefox git profile'
git pull origin master
else
logg info 'Cloning the public Firefox git profile'
cd " $ SETTINGS_DIR "
git clone {{ .firefoxPublicProfile }} profile.git
2023-01-29 23:27:31 -08:00
fi
2023-02-01 05:43:29 -08:00
2023-03-27 02:56:38 -07:00
### Copy user.js to profile.git profile
cp -f " ${ XDG_CONFIG_HOME : - $ HOME / . config } /firefox/user.js" " $ SETTINGS_DIR /profile.git"
### Git profile w/ plugins installed (installation happens below)
if [ ! -d " $ SETTINGS_DIR /profile.plugins" ]; then
logg info "Syncing $ SETTINGS_DIR /profile.git to $ SETTINGS_DIR /profile.plugins"
rsync -a " $ SETTINGS_DIR /profile.git/" " $ SETTINGS_DIR /profile.plugins"
rsync -a " ${ XDG_DATA_HOME : - $ HOME / . local / share } /firefox/" " $ SETTINGS_DIR /profile.plugins"
cp -f " ${ XDG_CONFIG_HOME : - $ HOME / . config } /firefox/user.js" " $ SETTINGS_DIR /profile.plugins"
fi
{{- if and (stat (joinPath .host.home ".config" "age" "chezmoi.txt") (ne .firefoxPrivateProfile false)) }}
2023-02-01 05:43:29 -08:00
### Private hosted profile
if [ ! -d " $ SETTINGS_DIR /profile.private" ]; then
logg info 'Downloading the encrypted Firefox private profile'
2023-02-01 07:10:51 -08:00
cd " $ SETTINGS_DIR "
2023-02-01 05:43:29 -08:00
curl -sSL '{{ .firefoxPrivateProfile }}' -o profile.private.tar.gz.age
logg info 'Decrypting the Firefox private profile'
2023-02-01 07:10:51 -08:00
chezmoi decrypt profile.private.tar.gz.age > profile.private.tar.gz || EXIT_DECRYPT_CODE=$?
if [ -z " $ EXIT_DECRYPT_CODE " ]; then
rm -f profile.private.tar.gz.age
logg info 'Decompressing the Firefox private profile'
tar -xzf profile.private.tar.gz
logg success 'The Firefox private profile was successfully installed'
2023-03-27 02:56:38 -07:00
cp -f " ${ XDG_CONFIG_HOME : - $ HOME / . config } /firefox/user.js" " $ SETTINGS_DIR /profile.private"
logg info 'Copied ~/.config/firefox/user.js to profile.private profile'
2023-02-01 07:10:51 -08:00
else
logg error 'Failed to decrypt the private Firefox profile'
fi
2023-02-01 05:43:29 -08:00
fi
2023-02-01 07:10:51 -08:00
{{- end }}
2023-01-25 00:46:31 -08:00
### Install Firefox addons (using list declared in .chezmoidata.yaml)
2023-03-27 02:56:38 -07:00
for SETTINGS_PROFILE in "profile.plugins" "profile.private"; do
if [ -d " $ SETTINGS_DIR / $ SETTINGS_PROFILE " ]; then
for FIREFOX_PLUGIN in {{ list (.firefoxAddOns | toString | replace "[" "" | replace "]" "") | uniq | join " " }}; do
logg info "Processing the $ FIREFOX_PLUGIN Firefox add-on"
PLUGIN_HTML="$(mktemp)"
curl --silent "https://addons.mozilla.org/en-US/firefox/addon/ $ FIREFOX_PLUGIN /" > " $ PLUGIN_HTML "
PLUGIN_TMP="$(mktemp)"
cat " $ PLUGIN_HTML " | htmlq ' #redux - store - state ' | sed ' s / ^ < scri . * application \/ json " >// ' | sed ' s/< \ /script>$// ' > " $ PLUGIN_TMP "
PLUGIN_ID="$(jq '.addons.bySlug["'" $ FIREFOX_PLUGIN "'"]' " $ PLUGIN_TMP ")"
if [ " $ PLUGIN_ID " != 'null' ]; then
PLUGIN_FILE_ID="$(jq -r '.addons.byID["'" $ PLUGIN_ID "'"].guid' " $ PLUGIN_TMP ")"
if [ " $ PLUGIN_FILE_ID " != 'null' ]; then
PLUGIN_URL="$(cat " $ PLUGIN_HTML " | htmlq '.InstallButtonWrapper-download-link' --attribute href)"
PLUGIN_FILENAME=" ${ PLUGIN_FILE_ID } .xpi"
PLUGIN_FOLDER="$(echo " $ PLUGIN_FILENAME " | sed 's/.xpi$//')"
if [ ! -d " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions/ $ PLUGIN_FOLDER " ]; then
logg info 'Downloading add-on XPI file for '" $ PLUGIN_FILENAME "' ('" $ FIREFOX_PLUGIN "')'
if [ ! -d " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions" ]; then
mkdir -p " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions"
fi
curl -sSL " $ PLUGIN_URL " -o " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions/ $ PLUGIN_FILENAME "
# Unzipping like this causes Firefox to complain about unsigned plugins
# TODO - figure out how to headlessly enable the extensions in such a way that is compatible with Flatpak / Snap
# using the /usr/lib/firefox/distribution/policies.json works but this is not compatible with Flatpak / Snap out of the box
# it seems since they do not have access to the file system by default. Also, using the policies.json approach forces
# all Firefox profiles to use the same extensions. Ideally, we should find a way to enable the extensions scoped
# to the user profile.
# logg info 'Unzipping '" $ PLUGIN_FILENAME "' ('" $ FIREFOX_PLUGIN "')'
# unzip " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions/ $ PLUGIN_FILENAME " -d " $ SETTINGS_DIR / $ SETTINGS_PROFILE /extensions/ $ PLUGIN_FOLDER "
logg success 'Installed `'" $ FIREFOX_PLUGIN "'`'
fi
else
logg warn 'A null Firefox add-on filename was detected for `'" $ FIREFOX_PLUGIN "'`'
2023-01-29 23:27:31 -08:00
fi
2023-03-27 02:56:38 -07:00
else
logg warn 'A null Firefox add-on ID was detected for `'" $ FIREFOX_PLUGIN "'`'
2023-01-25 00:46:31 -08:00
fi
2023-03-27 02:56:38 -07:00
done
2023-01-14 02:14:43 -08:00
fi
2023-01-25 00:46:31 -08:00
done
fi
done
2023-01-14 02:14:43 -08:00
{{ end -}}