#!/bin/sh set -eu DOTS_REPO="https://git.snaile.de/luca/dotfiles" DOTS_BRANCH="main" STOW_DIR=".local/share/stow" DOTS_PACKAGE="dots" CONFIG_FILE="./config.yml" USER_GROUPS="wheel,floppy,audio,video,cdrom,optical,kvm,xbuilder,users,docker" # Comma separated list SCRIPT_DIR="$(dirname "$(readlink -f "$0")")" BOLD="$(tput bold)" RED="$(tput setaf 1)" GREEN="$(tput setaf 2)" YELLOW="$(tput setaf 3)" BLUE="$(tput setaf 4)" RESET="$(tput sgr0)" error() { printf "%b\n" "${RED}${BOLD}${1}${RESET}" >&2 exit 1 } prompt() { message=$1 printf "%b" "${BLUE}${message}${RESET}" >"$(tty)" # shellcheck disable=SC3037,SC2046 read -r x echo "$x" unset x } emphasize() { printf "%b\n" "${GREEN}${BOLD}${1}${RESET}" } info() { printf "%b\n" "${1}" } warn() { printf "%b\n" "${YELLOW}${BOLD}${1}${RESET}" } print_buffer() { buffer_size=$1 line_length=$2 for i in $(seq 1 "$((buffer_size))"); do # shellcheck disable=SC2016 if [ "$trim" = "y" ]; then template='%-${line_length}s\n' else template='%-${line_length}.${line_length}s\n' fi # shellcheck disable=SC2086 eval '[ -z "${buffer'"$i"':-}" ] || { printf "'"$template"'" "${buffer'$i'}" }' done } push_buffer() { line=$1 buffer_size=$2 for i in $(seq 1 $((buffer_size - 1))); do eval 'buffer'"$i"'="${buffer'$((i + 1))':-}"' done eval 'buffer'"$buffer_size"'=''''"$line"''''' } scroll() { buffer_size=$1 # int trim=${2:-y} # 'y' | 'n' line_length=$(tput cols) trap 'line_length=$(tput cols)' WINCH tput civis while read -r line; do push_buffer "$line" "$buffer_size" tput sc print_buffer "$buffer_size" "$line_length" tput rc done tput cnorm for i in $(seq 1 $((buffer_size))); do eval "unset buffer$i" done print_buffer "$buffer_size" "$line_length" trap - WINCH } check_root() { [ "$(id -u)" = "0" ] || error "This script needs root!" } setup() { info "Synchronizing XBPS index..." xbps-install -S xbps void-repo-nonfree >/dev/null 2>&1 || error "Failed to synchronize XBPS index! (Try manually running xbps-install -S)" if ! xbps-query ntp >/dev/null 2>&1; then info "Installing ntp..." progs="ntp jq yq git stow" # shellcheck disable=SC2086 xbps-install -y $progs >/dev/null 2>&1 for prog in $progs; do command -v "${prog}" 1>/dev/null 2>&1 || error "${prog} isn't installed even though it should be!" done info "Synchronizing time..." ntpdate "pool.ntp.org" >/dev/null 2>&1 || warn "Failed to synchronize time!" fi info "Done!" } install_packages() { IFS=' ' for p in $(yq -c '.packages.[]' <config.yml); do flags=$(echo "$p" | yq -r '.command[1:].[]') packages=$(echo "$p" | yq -r '.list[]') # shellcheck disable=SC2086 [ -n "${packages}" ] && echo $flags $packages | xargs "$(echo "$p" | yq -r '.command[0]')" done | scroll 7 info "Done!" } install_files() { ( cd "${SCRIPT_DIR}/files" || exit 1 find . -type f,l -exec install -Dm 644 -o root -g root "{}" "/{}" \; ) info "Done!" } create_user() { failed=false while ! echo "$username" | grep "^[a-z_][a-z0-9_-]*$" | grep -qv "root"; do $failed && warn "Invalid username, try again!" username=$(prompt "Input Username: ") failed=true done if id -u "$username" >/dev/null 2>&1; then warn "User \"$username\" already exists, Skipping user creation!" usermod -G "$USER_GROUPS" "$username" else info "Creating user \"$username\" with the following groups: \"$USER_GROUPS\"..." useradd -m -G "$USER_GROUPS" "$username" failed=false while [ -z "$pass1" ] || [ "$pass1" != "$pass2" ]; do $failed && warn "Passwords do not match or are empty, try again!" pass1=$(prompt "Input Password: ") pass2=$(prompt "Repeat Password: ") failed=true done echo "$username:$pass1" | chpasswd fi user_home=$(getent passwd "$username" | cut -d ':' -f 6) [ -z "$username" ] && error "\$username variable is empty, this script is bugged!" [ -z "$user_home" ] && error "\$user_home variable is empty, this script is bugged!" sudo -u "$username" [ -w "$user_home" ] || error "$username can't write to '$user_home'!" info "Done!" } create_directories() { #shellcheck disable=SC2016 directories=$(yq -c '[].directories' <${CONFIG_FILE}) counter=1 num_dirs=$(echo "$directories" | wc -l) for entry in $directories; do path=$(echo "$entry" | jq -r '.path') mode=$(echo "$entry" | jq -r '.mode') [ "$mode" = "null" ] && mode="0755" info "Creating directory ${counter} of ${num_dirs}: ~/${path}" [ -d "${user_home}/${path}" ] || sudo -u "$username" mkdir -m "${mode}" -p "${user_home}/${path}" counter=$((counter + 1)) done info "Done!" } install_dotfiles() { info "Cloning dotfiles..." mkdir -p "${user_home}/${STOW_DIR}" if [ ! -d "${user_home}/${STOW_DIR}/${DOTS_PACKAGE}/.git" ]; then if ! git -C "${user_home}/${STOW_DIR}" clone -q --recurse-submodules -b "$DOTS_BRANCH" "$DOTS_REPO" "$DOTS_PACKAGE"; then warn "Failed to clone dotfiles" return 1 fi fi info "Symlinking dotfiles..." if ! stow -d "$user_home/$STOW_DIR" -t "$user_home" "$DOTS_PACKAGE" 1>/dev/null 2>&1; then warn "Failed to symlink dotfiles" return 2 fi info "Done!" } select_keymap() { [ -L "${user_home}/.local/share/xkb/compiled/keymap" ] && return map="$(find "${user_home}/.local/share/xkb/compiled" -type f -printf "%f\n" | fzf --header="Select a keymap keymap:")" ln -s "$map" "${user_home}/.local/share/xkb/compiled/keymap" } enable_services() { services=$(yq -r '.services[]' <config.yml) counter=1 num_services=$(echo "$services" | wc -l) info "Removing old services..." for sv in /var/service/*; do echo "$services" | grep -qx "$(basename "$sv")" || rm "$sv" done info "Enabling services..." for service in $services; do info "Enabling service ${counter} of ${num_services}: ${service}" [ ! -L "/var/service/${service}" ] && ln -s "/etc/sv/${service}" "/var/service/" done info "Done!" } finalize() { gid=$(getent passwd "$username" | cut -d ':' -f 4) groupname=$(getent group "$gid" | cut -d ':' -f 1) info "Setting ownership of home directories..." chown -v "$username:$groupname" -R "$user_home" 2>&1 | scroll 7 info "Done!" } ### CONTROL FLOW BEGINS HERE ### trap 'tput cnorm; tput sgr0; exit' INT TERM EXIT check_root emphasize "-- Copying Files --" install_files # emphasize "-- Creating Symlinks --" # create_symlinks emphasize "-- Preparing Installation --" setup emphasize "-- Installing Packages --" install_packages username="$SUDO_USER" if [ -z "$username" ]; then emphasize "-- Creating User Account --" create_user else user_home=$(getent passwd "$username" | cut -d ':' -f 6) fi emphasize "-- Creating Standard Home Directories --" create_directories emphasize "-- Installing Dotfiles --" install_dotfiles select_keymap emphasize "-- Enabling Services --" enable_services emphasize "-- Finalizing Installation --" finalize emphasize "-- Installation Complete --"