2023-03-27 16:22:12 -07:00
#!/usr/bin/env bash
2023-04-15 21:10:10 -07:00
# @file Quick Start Provision Script
2023-03-27 16:22:12 -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.
# @description
# 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.
#
# **Note**: `https://install.doctor/start` points to this file.
#
# ## 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.
#
# | 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 |
#
# 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`).
#
# 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.
#
# ## Variables
#
# 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-05-29 19:46:47 -07: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
#
# 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.
#
# ## Links
#
# [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)
# @description Ensure Ubuntu / Debian run in `noninteractive` mode
export DEBIAN_FRONTEND=noninteractive
2023-06-24 00:54:01 -07:00
# @description Load default settings if it is in a CI setting
if [ -n " $ CI " ]; then
export HOST=" $ HOST "
export NO_RESTART=true
export HEADLESS_INSTALL=true
export SOFTWARE_GROUP="Full"
export FULL_NAME="Brian Zalewski"
export PRIMARY_EMAIL="help@megabyte.space"
2023-08-02 11:41:43 -07:00
export PUBLIC_SERVICES_DOMAIN="lab.megabyte.space"
2023-06-24 00:54:01 -07:00
export RESTRICTED_ENVIRONMENT=false
export WORK_ENVIRONMENT=false
fi
2023-08-08 13:27:50 -07:00
# @description Disconnect from WARP, if connected
if command -v warp-cli > /dev/null; then
if warp-cli status | grep 'Connected' > /dev/null; then
warp-cli disconnect && echo "Disconnected WARP to prevent conflicts"
fi
fi
2023-03-27 16:22:12 -07:00
# @description Detect `START_REPO` format and determine appropriate git address, otherwise use the master Install Doctor branch
2023-05-29 19:46:47 -07:00
if [ -z " $ START_REPO " ] && [ -z " $ REPO " ]; then
2023-05-31 19:15:55 -07:00
START_REPO="https://github.com/megabyte-labs/install.doctor.git"
2023-03-27 16:22:12 -07:00
else
2023-05-29 19:46:47 -07:00
if [ -n " $ REPO " ] && [ -z " $ START_REPO " ]; then
START_REPO=" $ REPO "
fi
2023-07-08 19:55:17 -07:00
if [[ " $ START_REPO " == *"/"* ]]; then
2023-05-29 19:46:47 -07:00
# Either full git address or GitHubUser/RepoName
if [[ " $ START_REPO " == *":"* ]] || [[ " $ START_REPO " == *"//"* ]]; then
START_REPO=" $ START_REPO "
2023-03-27 16:22:12 -07:00
else
2023-05-29 19:46:47 -07:00
START_REPO="https://github.com/ ${ START_REPO } .git"
2023-03-27 16:22:12 -07:00
fi
2023-05-29 19:46:47 -07:00
else
START_REPO="https://github.com/ $ START_REPO /install.doctor.git"
fi
2023-03-27 16:22:12 -07:00
fi
{{ include "partials" "logg" }}
# @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)
sudo -n true || SUDO_EXIT_CODE=$?
logg info 'Your user will temporarily be granted passwordless sudo for the duration of the script'
if [ -n " $ SUDO_EXIT_CODE " ]; then
2023-11-04 18:46:18 -07:00
logg info 'Press CTRL+C to bypass this prompt to either enter your password when needed or perform a non-privileged installation'
2023-03-27 16:22:12 -07:00
logg info 'Note: Non-privileged installations are not yet supported'
fi
# @description Add current user to /etc/sudoers so that headless automation is possible
if ! sudo cat /etc/sudoers | grep ' # TEMPORARY FOR INSTALL DOCTOR' > /dev/null; then
2023-07-14 22:43:02 -07:00
if [ -n " $ SUDO_PASSWORD " ]; then
printf '%s\n' " $ SUDO_PASSWORD " | sudo -p "" -S echo "$(whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR INSTALL DOCTOR" | sudo tee -a /etc/sudoers
else
echo "$(whoami) ALL=(ALL:ALL) NOPASSWD: ALL # TEMPORARY FOR INSTALL DOCTOR" | sudo tee -a /etc/sudoers
fi
2023-03-27 16:22:12 -07:00
fi
# @section Qubes dom0 Bootstrap
# @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
if command -v qubesctl > /dev/null; then
# @description Ensure sys-whonix is configured (for Qubes dom0)
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
}
# @description Ensure dom0 is updated
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
# @description Ensure sys-whonix is running
if ! qvm-check --running sys-whonix; then
qvm-start sys-whonix --skip-if-running
configureWizard > /dev/null
fi
# @description Ensure TemplateVMs are updated
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
# @description Ensure provisioning VM can run commands on any VM
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
# @description Create provisioning VM and initialize the provisioning process from there
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
# @description Restart the provisioning process with the same script but from the provisioning VM
qvm-run --pass-io " $ ANSIBLE_PROVISION_VM " 'curl -sSL https://install.doctor/start > ~/start.sh && bash ~/start.sh'
exit 0
fi
# @description Ensure basic system packages are available on the device
{{ include "partials" "basic-deps" }}
# @description Ensure Homebrew is installed and available
{{ include "partials" "homebrew" }}
# @description Ensure Chezmoi is installed
if ! command -v chezmoi > /dev/null; then
brew install chezmoi
fi
# @description Ensure Node.js is installed
if ! command -v node > /dev/null; then
brew install node
fi
# @description Ensure ZX is installed
if ! command -v zx > /dev/null; then
brew install zx
fi
# @description Install Glow / Gum if the `HEADLESS_INSTALL` variable is not set to true
if [ " $ HEADLESS_INSTALL " != 'true' ]; then
# @description Ensure Gum is installed
if ! command -v gum > /dev/null; then
brew install gum
fi
# @description Ensure Glow is installed
if ! command -v glow > /dev/null; then
brew install glow
fi
fi
# @description Ensure the ${ XDG_DATA_HOME : - $ HOME / . local / share } /chezmoi directory is cloned and up-to-date
2023-11-03 22:57:09 -07:00
logg info 'Setting git http.postBuffer value high for large source repository'
git config --global http.postBuffer 524288000
2023-03-27 16:22:12 -07:00
if [ -d " ${ XDG_DATA_HOME : - $ HOME / . local / share } /chezmoi/.git" ]; then
cd " ${ XDG_DATA_HOME : - $ HOME / . local / share } /chezmoi"
2023-05-31 19:15:55 -07:00
logg info "Pulling the latest changes from ${ START_REPO : - https : / / github . com / megabyte - labs / install . doctor . git } "
2023-03-27 16:22:12 -07:00
git pull origin master
else
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
# @description If the ` ${ XDG_CONFIG_HOME : - $ HOME / . config } /chezmoi/chezmoi.yaml` file is missing, then guide the user through the initial setup
if [ ! -f " ${ XDG_CONFIG_HOME : - $ HOME / . config } /chezmoi/chezmoi.yaml" ]; then
# @description Show introduction message if Glow is installed
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:22:12 -07:00
fi
# @description Prompt for the software group if the `SOFTWARE_GROUP` variable is not defined
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-04-18 22:15:21 -07:00
SOFTWARE_GROUP="$(gum choose "General-Purpose-Server" "Basic" "Standard" "Full")"
2023-03-27 16:22:12 -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
# @description Run `chezmoi init` when the Chezmoi configuration is missing
2023-11-04 18:46:18 -07:00
logg info 'Running chezmoi init since the '" ${ XDG_CONFIG_HOME : - $ HOME / . config } /chezmoi/chezmoi.yaml"' is not present'
2023-03-27 16:22:12 -07:00
chezmoi init
fi
2023-05-29 19:46:47 -07:00
# @description Run `chezmoi apply` and enable verbose mode if the `DEBUG_MODE` or `DEBUG` environment variable is set to true
2023-11-04 18:46:18 -07:00
logg info 'Running chezmoi apply'
2023-03-27 16:22:12 -07:00
if [ " $ DEBUG_MODE " = 'true' ]; then
DEBUG_MODIFIER="-vvvvv"
2023-05-29 19:46:47 -07:00
elif [ " $ DEBUG " = 'true' ]; then
DEBUG_MODIFIER="-vvvvv"
2023-03-27 16:22:12 -07:00
fi
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-03-27 16:22:12 -07:00
# `--force` flag if the `HEADLESS_INSTALL` variable is set to true.
2023-08-17 22:12:36 -07:00
mkdir -p " $ HOME /.local/var/log/install.doctor"
2023-03-27 16:22:12 -07:00
if [ " $ HEADLESS_INSTALL " = 'true' ]; then
if command -v unbuffer > /dev/null; then
2023-08-17 22:12:36 -07:00
unbuffer -p chezmoi apply $ DEBUG_MODIFIER -k --force 2>&1 | tee " $ HOME /.local/var/log/install.doctor/install.doctor.$(date +%s).log"
2023-03-27 16:22:12 -07:00
else
2023-08-17 22:12:36 -07:00
chezmoi apply $ DEBUG_MODIFIER -k --force 2>&1 | tee " $ HOME /.local/var/log/install.doctor/install.doctor.$(date +%s).log"
2023-03-27 16:22:12 -07:00
fi
else
if command -v unbuffer > /dev/null; then
2023-08-17 22:12:36 -07:00
unbuffer -p chezmoi apply $ DEBUG_MODIFIER -k 2>&1 | tee " $ HOME /.local/var/log/install.doctor/install.doctor.$(date +%s).log"
2023-03-27 16:22:12 -07:00
else
2023-08-17 22:12:36 -07:00
chezmoi apply $ DEBUG_MODIFIER -k 2>&1 | tee " $ HOME /.local/var/log/install.doctor/install.doctor.$(date +%s).log"
2023-03-27 16:22:12 -07:00
fi
fi
# @description Ensure gsed is available on macOS (for modifying `/etc/sudoers` to remove passwordless sudo)
if [ -n " $ REMOVE_TMP_SUDOERS " ] && [ -d /Applications ] && [ -d /System ]; then
if ! command -v gsed > /dev/null; then
if command -v brew > /dev/null; then
brew install gsed
else
logg warn 'Homebrew is not available and passwordless sudo might still be enabled in /etc/sudoers. Modify the file manually if you wish to disable passwordless sudo.'
fi
fi
fi
# @description Ensure temporary passwordless sudo privileges are removed from `/etc/sudoers`
if command -v gsed > /dev/null; then
sudo gsed -i '/ # TEMPORARY FOR INSTALL DOCTOR/d' /etc/sudoers || logg warn 'Failed to remove passwordless sudo from the /etc/sudoers file'
else
sudo sed -i '/ # TEMPORARY FOR INSTALL DOCTOR/d' /etc/sudoers || logg warn 'Failed to remove passwordless sudo from the /etc/sudoers file'
fi
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-03-27 16:22:12 -07:00
logg success 'Provisioning complete!'
2023-08-07 22:29:21 -07:00
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"
2023-03-27 16:22:12 -07:00
fi