2023-03-27 16:22:12 -07:00
#!/usr/bin/env bash
2023-11-12 20:53:44 -08:00
# @file Quick Start Provision Script
2023-03-27 16:23:46 -07:00
# @brief Main entry point for Install Doctor that ensures Homebrew and a few dependencies are installed before cloning the repository and running Chezmoi.
2023-03-27 16:22:12 -07:00
# @description
2023-03-27 16:23:46 -07:00
# This script ensures Homebrew is installed and then installs a few dependencies that Install Doctor relies on.
# After setting up the minimal amount of changes required, it clones the Install Doctor repository (which you
# can customize the location of so you can use your own fork). It then proceeds by handing things over to
# Chezmoi which handles the dotfile application and synchronous scripts. Task is used in conjunction with
# Chezmoi to boost the performance in some spots by introducing asynchronous features.
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# **Note**: `https://install.doctor/start` points to this file.
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# ## Dependencies
#
# The chart below shows the dependencies we rely on to get Install Doctor going. The dependencies that are bolded
# are mandatory. The ones that are not bolded are conditionally installed only if they are required.
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# | Dependency | Description |
# |--------------------|--------------------------------------------------------------------------------------|
# | **Chezmoi** | Dotfile configuration manager (on-device provisioning) |
# | **Task** | Task runner used on-device for task parallelization and dependency management |
# | **ZX / Node.js** | ZX is a Node.js abstraction that allows for better scripts |
# | Gum | Gum is a terminal UI prompt CLI (which allows sweet, interactive prompts) |
# | Glow | Glow is a markdown renderer used for applying terminal-friendly styled to markdown |
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# There are also a handful of system packages that are installed like `curl` and `git`. Then, during the Chezmoi provisioning
# process, there are a handful of system packages that are installed to ensure things run smoothly. You can find more details
# about these extra system packages by browsing through the `home/.chezmoiscripts/${DISTRO_ID}/` folder and other applicable
# folders (e.g. `universal`).
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# Although Install Doctor comes with presets that install a whole gigantic amount of software, it can actually
# be quite good at provisioning minimal server environments where you want to keep the binaries to a minimum.
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# ## Variables
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# Specify certain environment variables to customize the behavior of Install Doctor. With the right combination of
# environment variables, this script can be run completely headlessly. This allows us to do things like test our
# provisioning script on a wide variety of operating systems.
2023-03-27 16:22:12 -07:00
#
2023-11-12 20:53:44 -08:00
# | Variable | Description |
# |---------------------------|-----------------------------------------------------------------------------------|
# | `START_REPO` (or `REPO`) | Variable to specify the Git fork to use when provisioning |
# | `ANSIBLE_PROVISION_VM` | **For Qubes**, determines the name of the VM used to provision the system |
# | `DEBUG_MODE` (or `DEBUG`) | Set to true to enable verbose logging |
2023-03-27 16:22:12 -07:00
#
2023-03-27 16:23:46 -07:00
# For a full list of variables you can use to customize Install Doctor, check out our [Customization](https://install.doctor/docs/customization)
# and [Secrets](https://install.doctor/docs/customization/secrets) documentation.
2023-03-27 16:22:12 -07:00
#
# ## Links
#
2023-03-27 16:23:46 -07:00
# [Install Doctor homepage](https://install.doctor)
# [Install Doctor documentation portal](https://install.doctor/docs) (includes tips, tricks, and guides on how to customize the system to your liking)
2023-11-27 14:53:29 -08:00
# @description This function logs with style using Gum if it is installed, otherwise it uses `echo`. It is also capable of leveraging Glow to render markdown.
# When Glow is not installed, it uses `cat`. The following sub-commands are available:
#
# | Sub-Command | Description |
# |-------------|-----------------------------------------------------------------------------------------------------|
# | `error` | Logs a bright red error message |
# | `info` | Logs a regular informational message |
# | `md` | Tries to render the specified file using `glow` if it is installed and uses `cat` as a fallback |
# | `prompt` | Alternative that logs a message intended to describe an upcoming user input prompt |
# | `star` | Alternative that logs a message that starts with a star icon |
# | `start` | Same as `success` |
# | `success` | Logs a success message that starts with green checkmark |
# | `warn` | Logs a bright yellow warning message |
2023-11-12 20:53:44 -08:00
# @example
# logger info "An informative log"
2023-11-27 14:53:29 -08:00
# @example
# logger md ~/README.md
2023-03-27 16:23:46 -07:00
logg( ) {
TYPE = " $1 "
MSG = " $2 "
if [ " $TYPE " = = 'error' ] ; then
if command -v gum > /dev/null; then
gum style --border= "thick" " $( gum style --foreground= "#ff0000" "✖" ) $( gum style --bold --background= "#ff0000" --foreground= "#ffffff" " ERROR " ) $( gum style --bold " $MSG " ) "
else
echo " ERROR: $MSG "
fi
elif [ " $TYPE " = = 'info' ] ; then
if command -v gum > /dev/null; then
2023-11-12 20:53:44 -08:00
gum style " $( gum style --foreground= "#00ffff" "○" ) $( gum style --faint " $MSG " ) "
2023-03-27 16:23:46 -07:00
else
echo " INFO: $MSG "
fi
elif [ " $TYPE " = = 'md' ] ; then
if command -v glow > /dev/null; then
glow " $MSG "
else
cat " $MSG "
fi
elif [ " $TYPE " = = 'prompt' ] ; then
if command -v gum > /dev/null; then
gum style " $( gum style --foreground= "#00008b" "▶" ) $( gum style --bold " $MSG " ) "
else
echo " PROMPT: $MSG "
fi
elif [ " $TYPE " = = 'star' ] ; then
if command -v gum > /dev/null; then
gum style " $( gum style --foreground= "#d1d100" "◆" ) $( gum style --bold " $MSG " ) "
else
echo " STAR: $MSG "
fi
elif [ " $TYPE " = = 'start' ] ; then
if command -v gum > /dev/null; then
gum style " $( gum style --foreground= "#00ff00" "▶" ) $( gum style --bold " $MSG " ) "
else
echo " START: $MSG "
fi
elif [ " $TYPE " = = 'success' ] ; then
if command -v gum > /dev/null; then
2023-11-27 15:12:15 -08:00
gum style " $( gum style --foreground= "#00ff00" "✔" ) $( gum style --bold " $MSG " ) "
2023-03-27 16:23:46 -07:00
else
echo " SUCCESS: $MSG "
fi
elif [ " $TYPE " = = 'warn' ] ; then
if command -v gum > /dev/null; then
gum style " $( gum style --foreground= "#d1d100" "◆" ) $( gum style --bold --background= "#ffff00" --foreground= "#000000" " WARNING " ) $( gum style --bold " $MSG " ) "
else
echo " WARNING: $MSG "
fi
else
if command -v gum > /dev/null; then
gum style " $( gum style --foreground= "#00ff00" "▶" ) $( gum style --bold " $TYPE " ) "
else
echo " $MSG "
fi
fi
}
2023-11-27 14:53:29 -08:00
2023-11-12 20:53:44 -08:00
# @section Environment variables and system dependencies
# @description Ensure Ubuntu / Debian run in `noninteractive` mode. Detect `START_REPO` format and determine appropriate git address,
# otherwise use the master Install Doctor branch
setEnvironmentVariables( ) {
export DEBIAN_FRONTEND = noninteractive
2023-11-12 22:32:18 -08:00
export HOMEBREW_NO_ENV_HINTS = true
2023-11-12 20:53:44 -08:00
if [ -z " $START_REPO " ] && [ -z " $REPO " ] ; then
export START_REPO = "https://github.com/megabyte-labs/install.doctor.git"
else
if [ -n " $REPO " ] && [ -z " $START_REPO " ] ; then
export START_REPO = " $REPO "
fi
if [ [ " $START_REPO " = = *"/" * ] ] ; then
# Either full git address or GitHubUser/RepoName
if [ [ " $START_REPO " = = *":" * ] ] || [ [ " $START_REPO " = = *"//" * ] ] ; then
export START_REPO = " $START_REPO "
else
export START_REPO = " https://github.com/ ${ START_REPO } .git "
fi
else
export START_REPO = " https://github.com/ $START_REPO /install.doctor.git "
fi
fi
}
2023-11-27 14:53:29 -08:00
# @description This function ensures dependencies like `git` and `curl` are installed. More specifically, this function will:
#
# 1. Check if `curl`, `git`, `expect`, `rsync`, and `unbuffer` are on the system
# 2. If any of the above are missing, it will then use the appropriate system package manager to satisfy the requirements. *Note that some of the requirements are not scanned for in order to keep it simple and fast.*
# 3. On macOS, the official Xcode Command Line Tools are installed.
2023-11-12 20:53:44 -08:00
ensureBasicDeps( ) {
2023-11-29 23:18:43 -08:00
if ! command -v curl > /dev/null || ! command -v git > /dev/null || ! command -v expect > /dev/null || ! command -v rsync > /dev/null || ! command -v unbuffer > /dev/null; then
2023-11-27 14:53:29 -08:00
if command -v apt-get > /dev/null; then
### Debian / Ubuntu
logg info 'Running sudo apt-get update' && sudo apt-get update
logg info 'Running sudo apt-get install -y build-essential curl expect git rsync procps file' && sudo apt-get install -y build-essential curl expect git rsync procps file
elif command -v dnf > /dev/null; then
### Fedora
logg info 'Running sudo dnf groupinstall -y "Development Tools"' && sudo dnf groupinstall -y 'Development Tools'
logg info 'Running sudo dnf install -y curl expect git rsync procps-ng file' && sudo dnf install -y curl expect git rsync procps-ng file
elif command -v yum > /dev/null; then
### CentOS
logg info 'Running sudo yum groupinstall -y "Development Tools"' && sudo yum groupinstall -y 'Development Tools'
logg info 'Running sudo yum install -y curl expect git rsync procps-ng file' && sudo yum install -y curl expect git rsync procps-ng file
elif command -v pacman > /dev/null; then
### Archlinux
logg info 'Running sudo pacman update' && sudo pacman update
logg info 'Running sudo pacman -Syu base-devel curl expect git rsync procps-ng file' && sudo pacman -Syu base-devel curl expect git rsync procps-ng file
elif command -v zypper > /dev/null; then
### OpenSUSE
logg info 'Running sudo zypper install -yt pattern devel_basis' && sudo zypper install -yt pattern devel_basis
logg info 'Running sudo zypper install -y curl expect git rsync procps file' && sudo zypper install -y curl expect git rsync procps file
elif command -v apk > /dev/null; then
### Alpine
logg info 'Running sudo apk add build-base curl expect git rsync ruby procps file' && sudo apk add build-base curl expect git rsync ruby procps file
elif [ -d /Applications ] && [ -d /Library ] ; then
### macOS
logg info "Ensuring Xcode Command Line Tools are installed.."
if ! xcode-select -p >/dev/null 2>& 1; then
logg info "Command Line Tools for Xcode not found"
### This temporary file prompts the 'softwareupdate' utility to list the Command Line Tools
touch /tmp/.com.apple.dt.CommandLineTools.installondemand.in-progress;
XCODE_PKG = " $( softwareupdate -l | grep "\*.*Command Line" | tail -n 1 | sed 's/^[^C]* //' ) "
logg info "Installing from softwareupdate" && softwareupdate -i " $XCODE_PKG " && logg success " Successfully installed $XCODE_PKG "
fi
elif [ [ " $OSTYPE " = = 'cygwin' ] ] || [ [ " $OSTYPE " = = 'msys' ] ] || [ [ " $OSTYPE " = = 'win32' ] ] ; then
### Windows
logg info 'Running choco install -y curl expect git rsync' && choco install -y curl expect git rsync
elif command -v nix-env > /dev/null; then
### NixOS
logg warn "TODO - Add support for NixOS"
elif [ [ " $OSTYPE " = = 'freebsd' * ] ] ; then
### FreeBSD
logg warn "TODO - Add support for FreeBSD"
elif command -v pkg > /dev/null; then
### Termux
logg warn "TODO - Add support for Termux"
elif command -v xbps-install > /dev/null; then
### Void
logg warn "TODO - Add support for Void"
2023-11-12 21:20:10 -08:00
fi
2023-11-12 20:53:44 -08:00
fi
2023-11-12 21:38:41 -08:00
}
2023-11-12 20:53:44 -08:00
2023-11-27 14:53:29 -08:00
# @description This function ensures Homebrew is installed and available in the `PATH`. It handles the installation of Homebrew on both **Linux and macOS**.
# It will attempt to bypass sudo password entry if it detects that it can do so. The function also has some error handling in regards to various
# directories falling out of the correct ownership and permission states. Finally, it loads Homebrew into the active profile (allowing other parts of the script
# to use the `brew` command).
#
# With Homebrew installed and available, the script finishes by installing the `gcc` Homebrew package which is a very common dependency.
2023-11-12 20:53:44 -08:00
ensureHomebrew( ) {
if ! command -v brew > /dev/null; then
2023-11-27 14:53:29 -08:00
if [ -d /home/linuxbrew/.linuxbrew/bin ] ; then
logg info "Sourcing from /home/linuxbrew/.linuxbrew/bin/brew" && eval " $( /home/linuxbrew/.linuxbrew/bin/brew shellenv) "
if ! command -v brew > /dev/null; then
logg error "The /home/linuxbrew/.linuxbrew directory exists but something is not right. Try removing it and running the script again." && exit 1
fi
elif [ -d " $HOME /.linuxbrew " ] ; then
logg info " Sourcing from $HOME /.linuxbrew/bin/brew " && eval " $( $HOME /.linuxbrew/bin/brew shellenv) "
if ! command -v brew > /dev/null; then
logg error " The $HOME /.linuxbrew directory exists but something is not right. Try removing it and running the script again. " && exit 1
fi
2023-11-12 20:53:44 -08:00
else
2023-11-27 14:53:29 -08:00
### Installs Homebrew and addresses a couple potential issues
if command -v sudo > /dev/null && sudo -n true; then
logg info "Installing Homebrew"
2023-12-05 11:40:25 -08:00
echo | /bin/bash -c " $( curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh) "
2023-11-27 14:53:29 -08:00
else
logg info "Homebrew is not installed. The script will attempt to install Homebrew and you might be prompted for your password."
2023-12-05 11:40:25 -08:00
/bin/bash -c " $( curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh) " || BREW_EXIT_CODE = " $? "
2023-11-27 14:53:29 -08:00
if [ -n " $BREW_EXIT_CODE " ] ; then
if command -v brew > /dev/null; then
logg warn "Homebrew was installed but part of the installation failed. Trying a few things to fix the installation.."
BREW_DIRS = "share/man share/doc share/zsh/site-functions etc/bash_completion.d"
for BREW_DIR in $BREW_DIRS ; do
if [ -d " $( brew --prefix) / $BREW_DIR " ] ; then
logg info " Chowning $( brew --prefix) / $BREW_DIR " && sudo chown -R " $( whoami) " " $( brew --prefix) / $BREW_DIR "
fi
done
logg info "Running brew update --force --quiet" && brew update --force --quiet && logg success "Successfully ran brew update --force --quiet"
fi
2023-11-12 20:53:44 -08:00
fi
fi
2023-03-27 16:23:46 -07:00
2023-11-27 14:53:29 -08:00
### Ensures the `brew` binary is available on Linux machines. macOS installs `brew` into the default `PATH` so nothing needs to be done for macOS.
if [ -d /home/linuxbrew/.linuxbrew/bin ] ; then
logg info "Sourcing shellenv from /home/linuxbrew/.linuxbrew/bin/brew" && eval " $( /home/linuxbrew/.linuxbrew/bin/brew shellenv) "
2023-12-05 11:40:25 -08:00
elif [ -f /usr/local/bin/brew ] ; then
logg info "Sourcing shellenv from /usr/local/bin/brew" && eval " $( /usr/local/bin/brew shellenv) "
elif [ -f " ${ HOMEBREW_PREFIX :- /opt/homebrew } /bin/brew " ] ; then
logg info "Sourcing shellenv from " ${ HOMEBREW_PREFIX :- /opt/homebrew } /bin/brew"" && eval " $( " ${ HOMEBREW_PREFIX :- /opt/homebrew } /bin/brew " shellenv) "
2023-11-27 14:53:29 -08:00
fi
2023-11-12 20:53:44 -08:00
fi
fi
2023-03-27 16:23:46 -07:00
2023-11-27 14:53:29 -08:00
### Ensure GCC is installed via Homebrew
if command -v brew > /dev/null; then
if ! brew list | grep gcc > /dev/null; then
2023-12-05 11:40:25 -08:00
logg info "Installing Homebrew gcc" && brew install --quiet gcc
2023-11-27 14:53:29 -08:00
fi
else
logg error "Failed to initialize Homebrew" && exit 2
2023-07-14 22:43:02 -07:00
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-12-05 11:40:25 -08:00
# @description This function determines whether or not a reboot is required on the target system.
# On Linux, it will check for the presence of the `/var/run/reboot-required` file to determine
# whether or not a reboot is required. On macOS, it will reboot `/Library/Updates/index.plist`
# to determine whether or not a reboot is required.
#
# After determining whether or not a reboot is required, the script will attempt to automatically
# reboot the machine.
handleRequiredReboot( ) {
if [ -d /Applications ] && [ -d /System ] ; then
### macOS
logg info 'Checking if there is a pending update' && defaults read /Library/Updates/index.plist InstallAtLogout
# TODO - Uncomment this when we can determine conditions for reboot
# sudo shutdown -r now
elif [ -f /var/run/reboot-required ] ; then
### Linux
logg info '/var/run/reboot-required is present so a reboot is required'
if command -v systemctl > /dev/null; then
logg info 'systemctl present so rebooting with sudo systemctl start reboot.target' && sudo systemctl start reboot.target
elif command -v reboot > /dev/null; then
logg info 'reboot available as command so rebooting with sudo reboot' && sudo reboot
elif command -v shutdown > /dev/null; then
logg info 'shutdown command available so rebooting with sudo shutdown -r now' && sudo shutdown -r now
else
logg warn 'Reboot required but unable to determine appropriate restart command'
fi
fi
}
2023-11-12 20:53:44 -08:00
# @description Load default settings if it is in a CI setting
setCIEnvironmentVariables( ) {
2023-12-06 22:15:21 -08:00
if [ -n " $CI " ] || [ -n " $TEST_INSTALL " ] ; then
2023-11-12 20:53:44 -08:00
logg info "Automatically setting environment variables since the CI environment variable is defined"
logg info "Setting NO_RESTART to true" && export NO_RESTART = true
logg info "Setting HEADLESS_INSTALL to true " && export HEADLESS_INSTALL = true
logg info "Setting SOFTWARE_GROUP to Full-Desktop" && export SOFTWARE_GROUP = "Full-Desktop"
logg info "Setting FULL_NAME to Brian Zalewski" && export FULL_NAME = "Brian Zalewski"
2023-12-06 22:15:21 -08:00
logg info "Setting PRIMARY_EMAIL to brian@megabyte.space" && export PRIMARY_EMAIL = "brian@megabyte.space"
2023-11-12 20:53:44 -08:00
logg info "Setting PUBLIC_SERVICES_DOMAIN to lab.megabyte.space" && export PUBLIC_SERVICES_DOMAIN = "lab.megabyte.space"
logg info "Setting RESTRICTED_ENVIRONMENT to false" && export RESTRICTED_ENVIRONMENT = false
logg info "Setting WORK_ENVIRONMENT to false" && export WORK_ENVIRONMENT = false
2023-12-05 23:58:50 -08:00
logg info " Setting HOST to $( hostname -s) " && export HOST = " $( hostname -s) "
2023-11-12 20:53:44 -08:00
fi
}
# @description Disconnect from WARP, if connected
ensureWarpDisconnected( ) {
if command -v warp-cli > /dev/null; then
if warp-cli status | grep 'Connected' > /dev/null; then
logg info "Disconnecting from WARP" && warp-cli disconnect && logg success "Disconnected WARP to prevent conflicts"
fi
fi
}
# @description Notify user that they can press CTRL+C to prevent `/etc/sudoers` from being modified (which is currently required for headless installs on some systems).
# Additionally, this function will add the current user to `/etc/sudoers` so that headless automation is possible.
setupPasswordlessSudo( ) {
sudo -n true || SUDO_EXIT_CODE = $?
logg info 'Your user will temporarily be granted passwordless sudo for the duration of the script'
2023-12-06 22:15:21 -08:00
if [ -n " $SUDO_EXIT_CODE " ] && [ -z " $SUDO_PASSWORD " ] && command -v chezmoi > /dev/null && [ -f " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/home/.chezmoitemplates/secrets/SUDO_PASSWORD " ] ; then
SUDO_PASSWORD = " $( chezmoi decrypt " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/home/.chezmoitemplates/secrets/SUDO_PASSWORD " ) "
export SUDO_PASSWORD
2023-11-12 20:53:44 -08:00
fi
2023-11-27 17:10:05 -08:00
if [ -n " $SUDO_PASSWORD " ] ; then
2023-11-29 23:18:43 -08:00
printf '%s\n' " $SUDO_PASSWORD " | sudo -p "" -S echo " $( whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR INSTALL DOCTOR " | sudo tee -a /etc/sudoers > /dev/null
2023-11-27 17:10:05 -08:00
else
2023-12-06 22:15:21 -08:00
logg info 'Press CTRL+C to bypass this prompt to either enter your password when needed or perform a non-privileged installation'
logg info 'Note: Non-privileged installations are not yet supported'
2023-11-29 23:18:43 -08:00
echo " $( whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR INSTALL DOCTOR " | sudo tee -a /etc/sudoers > /dev/null
2023-11-12 20:53:44 -08:00
fi
}
# @section Qubes dom0
# @description Ensure sys-whonix is configured (for Qubes dom0)
ensureSysWhonix( ) {
2023-03-27 16:23:46 -07:00
CONFIG_WIZARD_COUNT = 0
function configureWizard( ) {
if xwininfo -root -tree | grep "Anon Connection Wizard" ; then
WINDOW_ID = " $( xwininfo -root -tree | grep "Anon Connection Wizard" | sed 's/^ *\([^ ]*\) .*/\1/' ) "
xdotool windowactivate " $WINDOW_ID " && sleep 1 && xdotool key 'Enter' && sleep 1 && xdotool key 'Tab Tab Enter' && sleep 24 && xdotool windowactivate " $WINDOW_ID " && sleep 1 && xdotool key 'Enter' && sleep 300
qvm-shutdown --wait sys-whonix
sleep 3
qvm-start sys-whonix
if xwininfo -root -tree | grep "systemcheck | Whonix" > /dev/null; then
WINDOW_ID_SYS_CHECK = " $( xwininfo -root -tree | grep "systemcheck | Whonix" | sed 's/^ *\([^ ]*\) .*/\1/' ) "
if xdotool windowactivate " $WINDOW_ID_SYS_CHECK " ; then
sleep 1
xdotool key 'Enter'
fi
fi
else
sleep 3
CONFIG_WIZARD_COUNT = $(( CONFIG_WIZARD_COUNT + 1 ))
if [ [ " $CONFIG_WIZARD_COUNT " = = '4' ] ] ; then
echo "The sys-whonix anon-connection-wizard utility did not open."
else
echo "Checking for anon-connection-wizard again.."
configureWizard
fi
fi
}
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Ensure dom0 is updated
ensureDom0Updated( ) {
2023-03-27 16:23:46 -07:00
if [ ! -f /root/dom0-updated ] ; then
sudo qubesctl --show-output state.sls update.qubes-dom0
sudo qubes-dom0-update --clean -y
touch /root/dom0-updated
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Ensure sys-whonix is running
ensureSysWhonixRunning( ) {
2023-03-27 16:23:46 -07:00
if ! qvm-check --running sys-whonix; then
qvm-start sys-whonix --skip-if-running
configureWizard > /dev/null
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Ensure TemplateVMs are updated
ensureTemplateVMsUpdated( ) {
2023-03-27 16:23:46 -07:00
if [ ! -f /root/templatevms-updated ] ; then
# timeout of 10 minutes is added here because the whonix-gw VM does not like to get updated
# with this method. Anyone know how to fix this?
sudo timeout 600 qubesctl --show-output --skip-dom0 --templates state.sls update.qubes-vm & > /dev/null || true
while read -r RESTART_VM; do
qvm-shutdown --wait " $RESTART_VM "
done < <( qvm-ls --all --no-spinner --fields= name,state | grep Running | grep -v sys-net | grep -v sys-firewall | grep -v sys-whonix | grep -v dom0 | awk '{print $1}' )
sudo touch /root/templatevms-updated
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Ensure provisioning VM can run commands on any VM
ensureProvisioningVMPermissions( ) {
2023-03-27 16:23:46 -07:00
echo "/bin/bash" | sudo tee /etc/qubes-rpc/qubes.VMShell
sudo chmod 755 /etc/qubes-rpc/qubes.VMShell
echo " ${ ANSIBLE_PROVISION_VM : =provision } " ' dom0 allow' | sudo tee /etc/qubes-rpc/policy/qubes.VMShell
echo " $ANSIBLE_PROVISION_VM " ' $anyvm allow' | sudo tee -a /etc/qubes-rpc/policy/qubes.VMShell
sudo chown " $( whoami) : $( whoami) " /etc/qubes-rpc/policy/qubes.VMShell
sudo chmod 644 /etc/qubes-rpc/policy/qubes.VMShell
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Create provisioning VM and initialize the provisioning process from there
createAndInitProvisionVM( ) {
2023-03-27 16:23:46 -07:00
qvm-create --label red --template debian-11 " $ANSIBLE_PROVISION_VM " & > /dev/null || true
qvm-volume extend " $ANSIBLE_PROVISION_VM :private " "40G"
if [ -f ~/.vaultpass ] ; then
qvm-run " $ANSIBLE_PROVISION_VM " 'rm -f ~/QubesIncoming/dom0/.vaultpass'
qvm-copy-to-vm " $ANSIBLE_PROVISION_VM " ~/.vaultpass
qvm-run " $ANSIBLE_PROVISION_VM " 'cp ~/QubesIncoming/dom0/.vaultpass ~/.vaultpass'
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Restart the provisioning process with the same script but from the provisioning VM
runStartScriptInProvisionVM( ) {
2023-03-27 16:23:46 -07:00
qvm-run --pass-io " $ANSIBLE_PROVISION_VM " 'curl -sSL https://install.doctor/start > ~/start.sh && bash ~/start.sh'
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Perform Qubes dom0 specific logic like updating system packages, setting up the Tor VM, updating TemplateVMs, and
# beginning the provisioning process using Ansible and an AppVM used to handle the provisioning process
handleQubesDom0( ) {
if command -v qubesctl > /dev/null; then
ensureSysWhonix
ensureDom0Updated
ensureSysWhonixRunning
ensureTemplateVMsUpdated
ensureProvisioningVMPermissions
createAndInitProvisionVM
runStartScriptInProvisionVM
exit 0
fi
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @section Homebrew dependencies
# @description Helper function used by [[ensureHomebrewDeps]] to ensure a Homebrew package is installed after
# first checking if it is already available on the system.
installBrewPackage( ) {
if ! command -v " $1 " > /dev/null; then
logg 'Installing ' " $1 " ''
2023-12-05 11:40:25 -08:00
brew install --quiet " $1 "
2023-11-12 20:53:44 -08:00
fi
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Installs various dependencies using Homebrew.
#
2023-12-06 22:15:21 -08:00
# 1. Ensures Glow, Gum, Chezmoi, Node.js, and ZX are installed.
# 2. If the system is macOS, then also install `gsed` and `coreutils`.
2023-11-12 20:53:44 -08:00
ensureHomebrewDeps( ) {
### Base dependencies
2023-12-06 22:15:21 -08:00
installBrewPackage "glow"
installBrewPackage "gum"
2023-11-12 20:53:44 -08:00
installBrewPackage "chezmoi"
installBrewPackage "node"
installBrewPackage "zx"
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
### macOS
if [ -d /Applications ] && [ -d /System ] ; then
2023-12-05 11:40:25 -08:00
installBrewPackage "expect"
2023-11-12 20:53:44 -08:00
installBrewPackage "gsed"
2023-11-12 22:32:18 -08:00
if ! command -v gtimeout > /dev/null; then
2023-12-05 11:40:25 -08:00
brew install --quiet coreutils
2023-11-12 22:32:18 -08:00
fi
2023-11-12 20:53:44 -08:00
fi
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @section Chezmoi
# @description Ensure the `${XDG_DATA_HOME:-$HOME/.local/share}/chezmoi` directory is cloned and up-to-date using the previously
# set `START_REPO` as the source repository.
cloneChezmoiSourceRepo( ) {
2023-12-04 22:16:18 -08:00
if ! git config --get --global http.postBuffer > /dev/null; then
logg info 'Setting git http.postBuffer value high for large source repository' && git config --global http.postBuffer 524288000
fi
2023-11-12 20:53:44 -08:00
if [ -d " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/.git " ] ; then
logg info " Changing directory to ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi " && cd " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi "
logg info " Pulling the latest changes in ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi " && git pull origin master
else
logg info " Ensuring ${ XDG_DATA_HOME :- $HOME /.local/share } is a folder " && mkdir -p " ${ XDG_DATA_HOME :- $HOME /.local/share } "
logg info " Cloning ${ START_REPO } to ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi " && git clone " ${ START_REPO } " " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi "
fi
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Guide the user through the initial setup by showing TUI introduction and accepting input through various prompts.
#
# 1. Show `chezmoi-intro.md` with `glow`
# 2. Prompt for the software group if the `SOFTWARE_GROUP` variable is not defined
# 3. Run `chezmoi init` when the Chezmoi configuration is missing (i.e. `${XDG_CONFIG_HOME:-$HOME/.config}/chezmoi/chezmoi.yaml`)
initChezmoiAndPrompt( ) {
### Show `chezmoi-intro.md` with `glow`
2023-03-27 16:23:46 -07:00
if command -v glow > /dev/null; then
2023-08-08 12:11:31 -07:00
glow " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/docs/terminal/chezmoi-intro.md "
2023-03-27 16:23:46 -07:00
fi
2023-11-12 20:53:44 -08:00
### Prompt for the software group if the `SOFTWARE_GROUP` variable is not defined
2023-03-27 16:23:46 -07:00
if command -v gum > /dev/null; then
if [ -z " $SOFTWARE_GROUP " ] ; then
logg prompt 'Select the software group you would like to install. If your environment is a macOS, Windows, or environment with the DISPLAY environment variable then desktop software will be installed too. The software groups are in the ' " ${ XDG_CONFIG_HOME :- $HOME /.config } /chezmoi/chezmoi.yaml " ' file.'
2023-11-12 20:53:44 -08:00
SOFTWARE_GROUP = " $( gum choose "Basic" "Server" "Standard" "Full" ) "
2023-03-27 16:23:46 -07:00
export SOFTWARE_GROUP
fi
else
logg error 'Woops! Gum needs to be installed for the guided installation. Try running brew install gum' && exit 1
fi
2023-11-12 20:53:44 -08:00
if [ ! -f " ${ XDG_CONFIG_HOME :- $HOME /.config } /chezmoi/chezmoi.yaml " ] ; then
### Run `chezmoi init` when the Chezmoi configuration is missing
logg info 'Running chezmoi init since the ' " ${ XDG_CONFIG_HOME :- $HOME /.config } /chezmoi/chezmoi.yaml " ' is not present'
chezmoi init
fi
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @description Run `chezmoi apply` and enable verbose mode if the `DEBUG_MODE` or `DEBUG` environment variable is set to true
configureDebugMode( ) {
if [ -n " $DEBUG_MODE " ] || [ -n " $DEBUG " ] ; then
logg info "Either DEBUG_MODE or DEBUG environment variables were set so Chezmoi will be run in debug mode"
export DEBUG_MODIFIER = "-vvvvv"
fi
}
2023-03-27 16:23:46 -07:00
2023-08-17 22:12:36 -07:00
# @description Save the log of the provision process to `$HOME/.local/var/log/install.doctor/install.doctor.$(date +%s).log` and add the Chezmoi
2023-11-12 20:53:44 -08:00
# `--force` flag if the `HEADLESS_INSTALL` variable is set to `true`.
runChezmoi( ) {
mkdir -p " $HOME /.local/var/log/install.doctor "
LOG_FILE = " $HOME /.local/var/log/install.doctor/install.doctor. $( date +%s) .log "
if [ " $HEADLESS_INSTALL " = 'true' ] ; then
logg info 'Running chezmoi apply forcefully'
if command -v unbuffer > /dev/null; then
if command -v caffeinate > /dev/null; then
2023-12-05 11:40:25 -08:00
caffeinate unbuffer -p chezmoi apply $DEBUG_MODIFIER -k --force 2>& 1 | tee " $LOG_FILE "
2023-11-12 20:53:44 -08:00
else
unbuffer -p chezmoi apply $DEBUG_MODIFIER -k --force 2>& 1 | tee " $LOG_FILE "
fi
2023-11-07 20:46:59 -08:00
else
2023-11-12 20:53:44 -08:00
if command -v caffeinate > /dev/null; then
caffeinate chezmoi apply $DEBUG_MODIFIER -k --force 2>& 1 | tee " $LOG_FILE "
else
chezmoi apply $DEBUG_MODIFIER -k --force 2>& 1 | tee " $LOG_FILE "
fi
2023-11-07 20:46:59 -08:00
fi
2023-03-27 16:23:46 -07:00
else
2023-11-12 20:53:44 -08:00
logg info 'Running chezmoi apply'
if command -v unbuffer > /dev/null; then
if command -v caffeinate > /dev/null; then
caffeinate unbuffer -p chezmoi apply $DEBUG_MODIFIER -k 2>& 1 | tee " $LOG_FILE "
else
unbuffer -p chezmoi apply $DEBUG_MODIFIER -k 2>& 1 | tee " $LOG_FILE "
fi
2023-03-27 16:23:46 -07:00
else
2023-11-12 20:53:44 -08:00
if command -v caffeinate > /dev/null; then
caffeinate chezmoi apply $DEBUG_MODIFIER -k 2>& 1 | tee " $LOG_FILE "
else
chezmoi apply $DEBUG_MODIFIER -k 2>& 1 | tee " $LOG_FILE "
fi
2023-03-27 16:23:46 -07:00
fi
fi
2023-11-12 20:53:44 -08:00
}
2023-03-27 16:23:46 -07:00
2023-11-12 20:53:44 -08:00
# @section Post-provision logic
2023-03-27 16:23:46 -07:00
# @description Ensure temporary passwordless sudo privileges are removed from `/etc/sudoers`
2023-11-12 20:53:44 -08:00
removePasswordlessSudo( ) {
if command -v gsed > /dev/null; then
2023-03-27 16:23:46 -07:00
sudo gsed -i '/# TEMPORARY FOR INSTALL DOCTOR/d' /etc/sudoers || logg warn 'Failed to remove passwordless sudo from the /etc/sudoers file'
2023-11-12 20:53:44 -08:00
else
2023-03-27 16:23:46 -07:00
sudo sed -i '/# TEMPORARY FOR INSTALL DOCTOR/d' /etc/sudoers || logg warn 'Failed to remove passwordless sudo from the /etc/sudoers file'
2023-11-12 20:53:44 -08:00
fi
}
2023-03-27 16:23:46 -07:00
2023-08-07 22:29:21 -07:00
# @description Render the `docs/terminal/post-install.md` file to the terminal at the end of the provisioning process
2023-11-12 20:53:44 -08:00
postProvision( ) {
logg success 'Provisioning complete!'
if command -v glow > /dev/null && [ -f " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/docs/terminal/post-install.md " ] ; then
glow " ${ XDG_DATA_HOME :- $HOME /.local/share } /chezmoi/docs/terminal/post-install.md "
fi
}
# @section Execution order
# @description The `provisionLogic` function is used to define the order of the script. All of the functions it relies on are defined
# above.
provisionLogic( ) {
logg info "Setting environment variables" && setEnvironmentVariables
logg info "Handling CI variables" && setCIEnvironmentVariables
logg info "Ensuring WARP is disconnected" && ensureWarpDisconnected
2023-11-27 14:53:29 -08:00
logg info "Applying passwordless sudo" && setupPasswordlessSudo
2023-11-12 20:53:44 -08:00
logg info "Ensuring system Homebrew dependencies are installed" && ensureBasicDeps
logg info "Ensuring Homebrew is available" && ensureHomebrew
2023-11-12 21:44:44 -08:00
logg info "Installing Homebrew packages" && ensureHomebrewDeps
2023-11-12 20:53:44 -08:00
logg info "Handling Qubes dom0 logic (if applicable)" && handleQubesDom0
logg info "Cloning / updating source repository" && cloneChezmoiSourceRepo
logg info "Handling pre-provision logic" && initChezmoiAndPrompt
logg info "Handling debug mode if DEBUG or DEBUG_MODE are defined" && configureDebugMode
logg info "Running the Chezmoi provisioning" && runChezmoi
logg info "Ensuring temporary passwordless sudo is removed" && removePasswordlessSudo
logg info "Handling post-provision logic" && postProvision
2023-12-05 11:40:25 -08:00
logg info "Determing whether or not reboot" && handleRequiredReboot
2023-11-12 20:53:44 -08:00
}
provisionLogic