#!/usr/bin/env bash set -euo pipefail info() { printf '\n==> %s\n' "$1" } warn() { printf 'WARN: %s\n' "$1" >&2 } append_if_missing() { local line="$1" local file="$2" touch "$file" grep -qxF "$line" "$file" || echo "$line" >>"$file" } command_exists() { command -v "$1" >/dev/null 2>&1 } prompt_yes_no() { local prompt="$1" local default="${2:-Y}" local reply="" while true; do if [[ "$default" == "Y" ]]; then read -r -p "$prompt [Y/n] " reply reply="${reply:-Y}" else read -r -p "$prompt [y/N] " reply reply="${reply:-N}" fi case "${reply,,}" in y|yes) return 0 ;; n|no) return 1 ;; esac done } prompt_value() { local prompt="$1" local default="${2:-}" local reply="" if [[ -n "$default" ]]; then read -r -p "$prompt [$default] " reply printf '%s' "${reply:-$default}" else read -r -p "$prompt " reply printf '%s' "$reply" fi } prompt_required_value() { local prompt="$1" local default="${2:-}" local value="" while true; do value="$(prompt_value "$prompt" "$default")" if [[ -n "$value" ]]; then printf '%s' "$value" return 0 fi done } prompt_passphrase() { local pass1="" local pass2="" while true; do read -r -s -p "Enter SSH key passphrase: " pass1 printf '\n' read -r -s -p "Confirm SSH key passphrase: " pass2 printf '\n' if [[ "$pass1" == "$pass2" ]]; then printf '%s' "$pass1" return 0 fi warn "Passphrases did not match. Try again." done } sanitize_slug() { local value="$1" value="${value,,}" value="${value// /-}" value="$(printf '%s' "$value" | tr -cd 'a-z0-9_-')" printf '%s' "$value" } extract_host_from_git_url() { local url="$1" if [[ "$url" =~ ^ssh://([^/@]+@)?([^/:]+) ]]; then printf '%s' "${BASH_REMATCH[2]}" return 0 fi if [[ "$url" =~ ^([^/@]+@)?([^:]+): ]]; then printf '%s' "${BASH_REMATCH[2]}" return 0 fi if [[ "$url" =~ ^https?://([^/]+) ]]; then printf '%s' "${BASH_REMATCH[1]}" return 0 fi return 1 } install_pacman_package() { local package="$1" sudo pacman -S --needed --noconfirm "$package" } install_yay() { if command_exists yay; then info "yay already installed" return 0 fi info "Installing yay" git clone https://aur.archlinux.org/yay.git "$TMP_DIR/yay" ( cd "$TMP_DIR/yay" makepkg -si --noconfirm ) } write_element_desktop_entry() { local desktop_file="$1" local display_name="$2" local exec_line="$3" mkdir -p "$HOME_DIR/.local/share/applications" cat >"$desktop_file" <>"$HOME_DIR/.ssh/known_hosts" 2>/dev/null || true fi fi if prompt_yes_no "Copy the public SSH key to the clipboard?" "Y"; then if command_exists wl-copy; then wl-copy <"$ssh_key_path.pub" info "Your public SSH key has been copied to the clipboard" else warn "wl-copy is not installed; printing the key instead" cat "$ssh_key_path.pub" fi fi printf '\nPublic key: %s.pub\n' "$ssh_key_path" fi if prompt_yes_no "Clone a dotfiles repository into ~/.config?" "Y"; then info "Cloning dotfiles" dotfiles_repo="$(prompt_required_value "Dotfiles repository URL:")" config_target="$(prompt_required_value "Target config directory:" "$HOME_DIR/.config")" if [[ -d "$config_target" ]]; then if prompt_yes_no "Back up the existing '$config_target' before replacing it?" "Y"; then config_backup="${config_target}.backup.$(date +%Y%m%d-%H%M%S)" mv "$config_target" "$config_backup" info "Existing config moved to $config_backup" elif prompt_yes_no "Delete the existing '$config_target' and replace it?" "N"; then rm -rf "$config_target" else warn "Skipping dotfiles clone because target already exists" dotfiles_repo="" fi fi if [[ -n "$dotfiles_repo" ]]; then git clone "$dotfiles_repo" "$config_target" if prompt_yes_no "Add the dotfiles git host to ~/.ssh/known_hosts?" "Y"; then if dotfiles_host="$(extract_host_from_git_url "$dotfiles_repo")"; then mkdir -p "$HOME_DIR/.ssh" touch "$HOME_DIR/.ssh/known_hosts" ssh-keyscan -H "$dotfiles_host" >>"$HOME_DIR/.ssh/known_hosts" 2>/dev/null || true else warn "Could not determine a host from '$dotfiles_repo'" fi fi fi fi if prompt_yes_no "Clone a wallpapers repository?" "Y"; then info "Cloning wallpapers" wallpapers_repo="$(prompt_required_value "Wallpapers repository URL:")" wallpaper_target="$(prompt_required_value "Wallpaper target directory:" "$HOME_DIR/.local/share/wallpapers")" mkdir -p "$(dirname "$wallpaper_target")" if [[ -d "$wallpaper_target" ]]; then if prompt_yes_no "Delete the existing wallpaper directory '$wallpaper_target' first?" "Y"; then rm -rf "$wallpaper_target" else warn "Skipping wallpapers clone because target already exists" wallpapers_repo="" fi fi if [[ -n "$wallpapers_repo" ]]; then git clone "$wallpapers_repo" "$wallpaper_target" fi fi if prompt_yes_no "Create two custom Element desktop launchers?" "Y"; then info "Creating Element desktop entries" element_name_default="$(prompt_required_value "Name for the default Element profile launcher:")" element_name_secondary="$(prompt_required_value "Name for the second Element profile launcher:")" element_slug_default="$(sanitize_slug "$element_name_default")" element_slug_secondary="$(sanitize_slug "$element_name_secondary")" element_profile_secondary="$(prompt_required_value "Secondary Element profile id:" "profile1")" write_element_desktop_entry \ "$HOME_DIR/.local/share/applications/element-${element_slug_default}.desktop" \ "$element_name_default" \ 'element-desktop --ozone-platform=x11' write_element_desktop_entry \ "$HOME_DIR/.local/share/applications/element-${element_slug_secondary}.desktop" \ "$element_name_secondary" \ "element-desktop --profile $element_profile_secondary" update-desktop-database "$HOME_DIR/.local/share/applications" >/dev/null 2>&1 || true fi if prompt_yes_no "Enable the sshd system service?" "Y"; then info "Enabling sshd" sudo systemctl enable --now sshd.service fi if prompt_yes_no "Configure SDDM autologin?" "N"; then info "Configuring SDDM autologin" autologin_user="$(prompt_required_value "Autologin user:")" autologin_session="$(prompt_required_value "Autologin session:" "hyprland-uwsm")" sudo mkdir -p /etc/sddm.conf.d sudo tee /etc/sddm.conf.d/autologin.conf >/dev/null <