Dotfiles update 2025-12-24 17:36

This commit is contained in:
Aaron D. Lee
2025-12-24 17:36:41 -05:00
parent b52a55b5e8
commit b12a5a7b0f
7 changed files with 1354 additions and 177 deletions

2
.ssh-profiles Normal file
View File

@@ -0,0 +1,2 @@
# SSH Connection Profiles
# Format: name|user@host|port|key_file|options|description

View File

@@ -2,16 +2,40 @@
# ============================================================================ # ============================================================================
# Dotfiles Health Check (Arch/CachyOS) # Dotfiles Health Check (Arch/CachyOS)
# ============================================================================ # ============================================================================
# Comprehensive health check with Arch-specific diagnostics
#
# Usage:
# dotfiles-doctor.sh # Run all checks
# dotfiles-doctor.sh --fix # Attempt automatic fixes
# dotfiles-doctor.sh --quick # Quick essential checks only
# ============================================================================
# Note: Not using set -e because arithmetic operations like ((var++)) # Note: Not using set -e because arithmetic operations can return non-zero
# return 1 when var was 0, which would cause premature exit
readonly DOTFILES_HOME="${DOTFILES_HOME:-$HOME/.dotfiles}" readonly DOTFILES_HOME="${DOTFILES_HOME:-$HOME/.dotfiles}"
readonly DOTFILES_VERSION="3.0.0" readonly DOTFILES_VERSION="3.1.0"
# Parse arguments
DO_FIX=false
QUICK_MODE=false
for arg in "$@"; do
case "$arg" in
--fix) DO_FIX=true ;;
--quick) QUICK_MODE=true ;;
--help|-h)
echo "Usage: dotfiles-doctor.sh [OPTIONS]"
echo ""
echo "Options:"
echo " --fix Attempt automatic fixes for issues"
echo " --quick Run quick essential checks only"
echo " --help Show this help"
exit 0
;;
esac
done
# Source shared colors # Source shared colors
source "$DOTFILES_HOME/zsh/lib/colors.zsh" 2>/dev/null || { source "$DOTFILES_HOME/zsh/lib/colors.zsh" 2>/dev/null || {
# Fallback if colors.zsh not found
DF_RED=$'\033[0;31m' DF_GREEN=$'\033[0;32m' DF_YELLOW=$'\033[1;33m' DF_RED=$'\033[0;31m' DF_GREEN=$'\033[0;32m' DF_YELLOW=$'\033[1;33m'
DF_BLUE=$'\033[0;34m' DF_CYAN=$'\033[0;36m' DF_NC=$'\033[0m' DF_BLUE=$'\033[0;34m' DF_CYAN=$'\033[0;36m' DF_NC=$'\033[0m'
DF_GREY=$'\033[38;5;242m' DF_LIGHT_BLUE=$'\033[38;5;39m' DF_GREY=$'\033[38;5;242m' DF_LIGHT_BLUE=$'\033[38;5;39m'
@@ -23,15 +47,13 @@ TOTAL_CHECKS=0
PASSED_CHECKS=0 PASSED_CHECKS=0
FAILED_CHECKS=0 FAILED_CHECKS=0
WARNING_CHECKS=0 WARNING_CHECKS=0
FIXED_CHECKS=0
# ============================================================================ # ============================================================================
# MOTD-style header # MOTD-style header
# ============================================================================ # ============================================================================
print_header() { print_header() {
if declare -f df_print_header &>/dev/null; then
df_print_header "dotfiles-doctor"
else
local user="${USER:-root}" local user="${USER:-root}"
local hostname="${HOSTNAME:-$(hostname -s 2>/dev/null)}" local hostname="${HOSTNAME:-$(hostname -s 2>/dev/null)}"
local datetime=$(date '+%a %b %d %H:%M') local datetime=$(date '+%a %b %d %H:%M')
@@ -40,10 +62,9 @@ print_header() {
echo "" echo ""
echo -e "${DF_GREY}${hline}${DF_NC}" echo -e "${DF_GREY}${hline}${DF_NC}"
echo -e "${DF_GREY}${DF_NC} ${DF_BOLD}${DF_LIGHT_BLUE}${user}@${hostname}${DF_NC} ${DF_DIM}dotfiles-doctor${DF_NC} ${datetime} ${DF_GREY}${DF_NC}" echo -e "${DF_GREY}${DF_NC} ${DF_BOLD}${DF_LIGHT_BLUE}${user}@${hostname}${DF_NC} ${DF_DIM}dotfiles-doctor v${DOTFILES_VERSION}${DF_NC} ${datetime} ${DF_GREY}${DF_NC}"
echo -e "${DF_GREY}${hline}${DF_NC}" echo -e "${DF_GREY}${hline}${DF_NC}"
echo "" echo ""
fi
} }
# ============================================================================ # ============================================================================
@@ -72,22 +93,42 @@ check_warn() {
echo -e " ${DF_YELLOW}${DF_NC} $1" echo -e " ${DF_YELLOW}${DF_NC} $1"
} }
check_fixed() {
FIXED_CHECKS=$((FIXED_CHECKS + 1))
echo -e " ${DF_CYAN}${DF_NC} Fixed: $1"
}
# ============================================================================ # ============================================================================
# Health checks # Core Health Checks
# ============================================================================ # ============================================================================
check_os() { check_os() {
print_section "Operating System" print_section "Operating System"
if [[ "$OSTYPE" == "linux-gnu" ]]; then if [[ "$OSTYPE" == "linux-gnu"* ]]; then
if grep -qi "arch\|cachyos" /etc/os-release 2>/dev/null; then if grep -qi "cachyos" /etc/os-release 2>/dev/null; then
check_pass "Running on Arch/CachyOS" local version=$(grep "VERSION_ID" /etc/os-release 2>/dev/null | cut -d= -f2 | tr -d '"')
check_pass "Running CachyOS ${version}"
elif grep -qi "arch" /etc/os-release 2>/dev/null; then
check_pass "Running Arch Linux"
else else
check_fail "Not running on Arch/CachyOS" check_fail "Not running on Arch/CachyOS"
fi fi
else else
check_fail "Not running on Linux" check_fail "Not running on Linux"
fi fi
# Kernel check
local kernel=$(uname -r)
if [[ "$kernel" == *"cachyos"* ]]; then
check_pass "CachyOS kernel: $kernel"
elif [[ "$kernel" == *"zen"* ]]; then
check_pass "Zen kernel: $kernel"
elif [[ "$kernel" == *"lts"* ]]; then
check_pass "LTS kernel: $kernel"
else
check_pass "Kernel: $kernel"
fi
} }
check_shell() { check_shell() {
@@ -97,12 +138,24 @@ check_shell() {
check_pass "Zsh configuration exists" check_pass "Zsh configuration exists"
else else
check_fail "Zsh configuration missing" check_fail "Zsh configuration missing"
if [[ "$DO_FIX" == true ]]; then
ln -sf "$DOTFILES_HOME/zsh/.zshrc" "$HOME/.zshrc" 2>/dev/null && check_fixed ".zshrc symlink created"
fi
fi fi
if [[ "$SHELL" == *"zsh"* ]]; then if [[ "$SHELL" == *"zsh"* ]]; then
check_pass "Zsh is default shell" check_pass "Zsh is default shell"
else else
check_warn "Zsh is not default shell (current: $SHELL)" check_warn "Zsh is not default shell (current: $SHELL)"
if [[ "$DO_FIX" == true ]]; then
echo " Run: chsh -s \$(which zsh)"
fi
fi
# Check if zsh is recent version
if command -v zsh &>/dev/null; then
local zsh_version=$(zsh --version | awk '{print $2}')
check_pass "Zsh version: $zsh_version"
fi fi
} }
@@ -114,13 +167,15 @@ check_symlinks() {
for symlink in ~/.zshrc ~/.gitconfig ~/.vimrc ~/.tmux.conf; do for symlink in ~/.zshrc ~/.gitconfig ~/.vimrc ~/.tmux.conf; do
if [[ -L "$symlink" ]]; then if [[ -L "$symlink" ]]; then
((symlink_count++)) symlink_count=$((symlink_count + 1))
if [[ -e "$symlink" ]]; then if [[ -e "$symlink" ]]; then
check_pass "$(basename $symlink)$(readlink $symlink)" check_pass "$(basename $symlink)$(readlink $symlink)"
else else
((broken_count++)) broken_count=$((broken_count + 1))
check_fail "$(basename $symlink) is broken" check_fail "$(basename $symlink) is broken"
fi fi
elif [[ -f "$symlink" ]]; then
check_warn "$(basename $symlink) is regular file (not symlink)"
fi fi
done done
@@ -133,14 +188,14 @@ check_vim() {
print_section "Editor Configuration" print_section "Editor Configuration"
if command -v vim &>/dev/null; then if command -v vim &>/dev/null; then
local vim_version=$(vim --version | head -1) local vim_version=$(vim --version | head -1 | awk '{print $5}')
check_pass "Vim installed: $vim_version" check_pass "Vim installed: $vim_version"
else else
check_fail "Vim not installed" check_fail "Vim not installed"
fi fi
if command -v nvim &>/dev/null; then if command -v nvim &>/dev/null; then
local nvim_version=$(nvim --version | head -1) local nvim_version=$(nvim --version | head -1 | awk '{print $2}')
check_pass "Neovim installed: $nvim_version" check_pass "Neovim installed: $nvim_version"
else else
check_warn "Neovim not installed (optional)" check_warn "Neovim not installed (optional)"
@@ -155,14 +210,13 @@ check_git() {
if git config --global user.name &>/dev/null; then if git config --global user.name &>/dev/null; then
local git_user=$(git config --global user.name) local git_user=$(git config --global user.name)
check_pass "Git user configured: $git_user" check_pass "Git user: $git_user"
else else
check_fail "Git user not configured" check_fail "Git user not configured"
fi fi
if git config --global user.email &>/dev/null; then if git config --global user.email &>/dev/null; then
local git_email=$(git config --global user.email) check_pass "Git email configured"
check_pass "Git email configured: $git_email"
else else
check_fail "Git email not configured" check_fail "Git email not configured"
fi fi
@@ -171,45 +225,9 @@ check_git() {
fi fi
} }
check_optional_tools() { # ============================================================================
print_section "Optional Tools" # Arch-Specific Checks
# ============================================================================
if command -v fzf &> /dev/null; then
check_pass "fzf installed (fuzzy finder)"
else
check_warn "fzf not installed (command palette requires this)"
fi
if command -v lastpass-cli &> /dev/null || command -v lpass &> /dev/null; then
check_pass "LastPass CLI installed"
else
check_warn "LastPass CLI not installed (password manager)"
fi
if command -v tmux &> /dev/null; then
check_pass "Tmux installed"
else
check_warn "Tmux not installed (workspaces require this)"
fi
if command -v age &> /dev/null || command -v gpg &> /dev/null; then
check_pass "Encryption tool available (age or gpg)"
else
check_warn "No encryption tool (vault requires age or gpg)"
fi
if command -v bat &> /dev/null; then
check_pass "bat installed (syntax highlighting)"
else
check_warn "bat not installed (optional enhancement)"
fi
if command -v eza &> /dev/null; then
check_pass "eza installed (ls replacement)"
else
check_warn "eza not installed (optional enhancement)"
fi
}
check_pacman() { check_pacman() {
print_section "Package Manager" print_section "Package Manager"
@@ -217,7 +235,166 @@ check_pacman() {
if command -v pacman &>/dev/null; then if command -v pacman &>/dev/null; then
check_pass "Pacman available" check_pass "Pacman available"
else else
check_fail "Pacman not found (this is Arch/CachyOS only)" check_fail "Pacman not found"
return
fi
# Check for AUR helper
if command -v paru &>/dev/null; then
check_pass "AUR helper: paru"
elif command -v yay &>/dev/null; then
check_pass "AUR helper: yay"
else
check_warn "No AUR helper installed (recommend: paru)"
fi
}
check_pacman_health() {
[[ "$QUICK_MODE" == true ]] && return
print_section "Pacman Health"
# Check for orphaned packages
local orphans=$(pacman -Qtdq 2>/dev/null | wc -l)
if [[ $orphans -eq 0 ]]; then
check_pass "No orphaned packages"
else
check_warn "$orphans orphaned package(s)"
if [[ "$DO_FIX" == true ]]; then
echo " Clean: pacman -Qtdq | sudo pacman -Rns -"
fi
fi
# Check package cache size
if [[ -d /var/cache/pacman/pkg ]]; then
local cache_size=$(du -sh /var/cache/pacman/pkg 2>/dev/null | cut -f1)
local pkg_count=$(ls /var/cache/pacman/pkg 2>/dev/null | wc -l)
if [[ $pkg_count -gt 500 ]]; then
check_warn "Package cache: $cache_size ($pkg_count files)"
if [[ "$DO_FIX" == true ]]; then
echo " Clean: sudo paccache -rk2"
fi
else
check_pass "Package cache: $cache_size"
fi
fi
# Check for available updates
if command -v checkupdates &>/dev/null; then
local updates=$(checkupdates 2>/dev/null | wc -l)
if [[ $updates -eq 0 ]]; then
check_pass "System up to date"
else
check_warn "$updates update(s) available"
fi
fi
}
check_systemd() {
[[ "$QUICK_MODE" == true ]] && return
print_section "Systemd Services"
# Check for failed services
local failed_count=$(systemctl --failed --no-pager --no-legend 2>/dev/null | wc -l)
if [[ $failed_count -eq 0 ]]; then
check_pass "No failed system services"
else
check_fail "$failed_count failed service(s)"
systemctl --failed --no-pager --no-legend 2>/dev/null | head -3 | while read -r line; do
local svc=$(echo "$line" | awk '{print $1}')
echo -e " ${DF_DIM}$svc${DF_NC}"
done
fi
# Check user services
local user_failed=$(systemctl --user --failed --no-pager --no-legend 2>/dev/null | wc -l)
if [[ $user_failed -eq 0 ]]; then
check_pass "No failed user services"
else
check_warn "$user_failed failed user service(s)"
fi
}
check_btrfs() {
[[ "$QUICK_MODE" == true ]] && return
# Only check if root is btrfs
local fstype=$(df -T / 2>/dev/null | awk 'NR==2 {print $2}')
[[ "$fstype" != "btrfs" ]] && return
print_section "Btrfs Filesystem"
check_pass "Root filesystem: btrfs"
# Check for device errors
local stats=$(sudo btrfs device stats / 2>/dev/null)
local errors=$(echo "$stats" | grep -v " 0$" | grep -v "^$")
if [[ -z "$errors" ]]; then
check_pass "No btrfs device errors"
else
check_fail "Btrfs errors detected!"
echo "$errors" | head -3 | while read -r line; do
echo -e " ${DF_DIM}$line${DF_NC}"
done
fi
# Check last scrub
local scrub_info=$(sudo btrfs scrub status / 2>/dev/null)
if echo "$scrub_info" | grep -q "running"; then
check_pass "Scrub currently running"
elif echo "$scrub_info" | grep -q "finished"; then
local scrub_date=$(echo "$scrub_info" | grep "Scrub started" | awk '{print $3, $4}')
check_pass "Last scrub: $scrub_date"
else
check_warn "No scrub history (recommend monthly)"
fi
# Check snapper
if command -v snapper &>/dev/null && [[ -d "/.snapshots" ]]; then
local snap_count=$(sudo snapper -c root list 2>/dev/null | tail -n +3 | wc -l)
check_pass "Snapper: $snap_count snapshot(s)"
fi
}
# ============================================================================
# Standard Checks
# ============================================================================
check_optional_tools() {
print_section "Optional Tools"
if command -v fzf &>/dev/null; then
check_pass "fzf (fuzzy finder)"
else
check_warn "fzf not installed (command palette needs this)"
fi
if command -v bat &>/dev/null; then
check_pass "bat (syntax highlighting)"
else
check_warn "bat not installed"
fi
if command -v eza &>/dev/null; then
check_pass "eza (modern ls)"
else
check_warn "eza not installed"
fi
if command -v tmux &>/dev/null; then
check_pass "tmux (terminal multiplexer)"
else
check_warn "tmux not installed"
fi
if command -v age &>/dev/null || command -v gpg &>/dev/null; then
check_pass "Encryption available (age/gpg)"
else
check_warn "No encryption tool (vault needs age/gpg)"
fi fi
} }
@@ -229,15 +406,23 @@ check_permissions() {
check_pass "install.sh is executable" check_pass "install.sh is executable"
else else
check_fail "install.sh is not executable" check_fail "install.sh is not executable"
if [[ "$DO_FIX" == true ]]; then
chmod +x "$DOTFILES_HOME/install.sh"
check_fixed "install.sh permissions"
fi
fi fi
fi fi
if [[ -d "$DOTFILES_HOME/bin" ]]; then if [[ -d "$DOTFILES_HOME/bin" ]]; then
local non_exec=$(find "$DOTFILES_HOME/bin" -type f ! -perm /u+x 2>/dev/null | wc -l) local non_exec=$(find "$DOTFILES_HOME/bin" -type f ! -perm /u+x 2>/dev/null | wc -l)
if [[ $non_exec -eq 0 ]]; then if [[ $non_exec -eq 0 ]]; then
check_pass "All scripts in bin/ are executable" check_pass "All bin/ scripts executable"
else else
check_fail "$non_exec scripts in bin/ are not executable" check_fail "$non_exec bin/ scripts not executable"
if [[ "$DO_FIX" == true ]]; then
find "$DOTFILES_HOME/bin" -type f ! -perm /u+x -exec chmod +x {} \;
check_fixed "bin/ permissions"
fi
fi fi
fi fi
} }
@@ -249,19 +434,19 @@ check_zsh_plugins() {
check_pass "Oh My Zsh installed" check_pass "Oh My Zsh installed"
if [[ -d "$HOME/.oh-my-zsh/custom/plugins/zsh-autosuggestions" ]]; then if [[ -d "$HOME/.oh-my-zsh/custom/plugins/zsh-autosuggestions" ]]; then
check_pass "zsh-autosuggestions installed" check_pass "zsh-autosuggestions"
else else
check_warn "zsh-autosuggestions not installed" check_warn "zsh-autosuggestions not installed"
fi fi
if [[ -d "$HOME/.oh-my-zsh/custom/plugins/zsh-syntax-highlighting" ]]; then if [[ -d "$HOME/.oh-my-zsh/custom/plugins/zsh-syntax-highlighting" ]]; then
check_pass "zsh-syntax-highlighting installed" check_pass "zsh-syntax-highlighting"
else else
check_warn "zsh-syntax-highlighting not installed" check_warn "zsh-syntax-highlighting not installed"
fi fi
if [[ -f "$HOME/.oh-my-zsh/themes/adlee.zsh-theme" ]]; then if [[ -f "$HOME/.oh-my-zsh/themes/adlee.zsh-theme" ]]; then
check_pass "adlee theme installed" check_pass "adlee theme"
else else
check_warn "adlee theme not installed" check_warn "adlee theme not installed"
fi fi
@@ -274,27 +459,33 @@ check_dotfiles_dir() {
print_section "Dotfiles Directory" print_section "Dotfiles Directory"
if [[ -d "$DOTFILES_HOME" ]]; then if [[ -d "$DOTFILES_HOME" ]]; then
check_pass "Dotfiles directory found: $DOTFILES_HOME" check_pass "Dotfiles: $DOTFILES_HOME"
else else
check_fail "Dotfiles directory not found: $DOTFILES_HOME" check_fail "Dotfiles not found: $DOTFILES_HOME"
return return
fi fi
if [[ -f "$DOTFILES_HOME/dotfiles.conf" ]]; then if [[ -f "$DOTFILES_HOME/dotfiles.conf" ]]; then
check_pass "Configuration file exists" check_pass "Config file exists"
else else
check_warn "Configuration file missing" check_warn "Config file missing"
fi fi
if [[ -d "$DOTFILES_HOME/.git" ]]; then if [[ -d "$DOTFILES_HOME/.git" ]]; then
check_pass "Git repository initialized" check_pass "Git repo initialized"
# Check for uncommitted changes
local changes=$(cd "$DOTFILES_HOME" && git status --porcelain 2>/dev/null | wc -l)
if [[ $changes -gt 0 ]]; then
check_warn "$changes uncommitted change(s)"
fi
else else
check_warn "Not a git repository" check_warn "Not a git repository"
fi fi
} }
# ============================================================================ # ============================================================================
# Print summary # Print Summary
# ============================================================================ # ============================================================================
print_summary() { print_summary() {
@@ -304,21 +495,29 @@ print_summary() {
if [[ $FAILED_CHECKS -eq 0 ]]; then if [[ $FAILED_CHECKS -eq 0 ]]; then
echo -e "${DF_GREEN}${DF_NC} All checks passed ($PASSED_CHECKS/$TOTAL_CHECKS)" echo -e "${DF_GREEN}${DF_NC} All checks passed ($PASSED_CHECKS/$TOTAL_CHECKS)"
else else
echo -e "${DF_RED}${DF_NC} Some checks failed" echo -e "${DF_RED}${DF_NC} Issues found"
echo -e " ${DF_GREEN}Passed:${DF_NC} $PASSED_CHECKS" echo -e " ${DF_GREEN}Passed:${DF_NC} $PASSED_CHECKS"
echo -e " ${DF_RED}Failed:${DF_NC} $FAILED_CHECKS" echo -e " ${DF_RED}Failed:${DF_NC} $FAILED_CHECKS"
if [[ $WARNING_CHECKS -gt 0 ]]; then if [[ $WARNING_CHECKS -gt 0 ]]; then
echo -e " ${DF_YELLOW}Warnings:${DF_NC} $WARNING_CHECKS" echo -e " ${DF_YELLOW}Warnings:${DF_NC} $WARNING_CHECKS"
fi fi
if [[ $FIXED_CHECKS -gt 0 ]]; then
echo -e " ${DF_CYAN}Fixed:${DF_NC} $FIXED_CHECKS"
fi
fi fi
echo "" echo ""
if [[ $FAILED_CHECKS -gt 0 ]]; then if [[ $FAILED_CHECKS -gt 0 && "$DO_FIX" != true ]]; then
echo -e "${DF_YELLOW}💡 Tip:${DF_NC} Run 'dotfiles-doctor.sh --fix' to attempt automatic fixes" echo -e "${DF_YELLOW}💡${DF_NC} Run with --fix to attempt automatic fixes"
echo "" echo ""
return 1 return 1
fi fi
if [[ $FIXED_CHECKS -gt 0 ]]; then
echo -e "${DF_CYAN}${DF_NC} Fixed $FIXED_CHECKS issue(s). Run again to verify."
echo ""
fi
} }
# ============================================================================ # ============================================================================
@@ -328,16 +527,24 @@ print_summary() {
main() { main() {
print_header print_header
# Essential checks (always run)
check_os check_os
check_pacman check_pacman
check_shell check_shell
check_vim
check_git
check_dotfiles_dir check_dotfiles_dir
check_symlinks check_symlinks
# Additional checks (skip in quick mode)
if [[ "$QUICK_MODE" != true ]]; then
check_vim
check_git
check_zsh_plugins check_zsh_plugins
check_optional_tools check_optional_tools
check_permissions check_permissions
check_pacman_health
check_systemd
check_btrfs
fi
print_summary print_summary
} }

View File

@@ -43,7 +43,7 @@ SET_ZSH_DEFAULT="ask"
# --- MOTD (Message of the Day) --- # --- MOTD (Message of the Day) ---
ENABLE_MOTD="true" ENABLE_MOTD="true"
MOTD_STYLE="compact" MOTD_STYLE="compact" # compact, mini, full, or none
# --- Theme Settings --- # --- Theme Settings ---
ZSH_THEME_NAME="adlee" ZSH_THEME_NAME="adlee"
@@ -77,6 +77,30 @@ SSH_AUTO_TMUX="true"
SSH_TMUX_SESSION_PREFIX="ssh" SSH_TMUX_SESSION_PREFIX="ssh"
SSH_SYNC_DOTFILES="ask" SSH_SYNC_DOTFILES="ask"
# ============================================================================
# Btrfs Helpers (btrfs-helpers.zsh)
# ============================================================================
# Default mount point for btrfs commands
BTRFS_DEFAULT_MOUNT="/"
# ============================================================================
# Systemd Helpers (systemd-helpers.zsh)
# ============================================================================
# Show failed services count in MOTD
MOTD_SHOW_FAILED_SERVICES="true"
# ============================================================================
# Package Manager
# ============================================================================
# Show available updates in MOTD
MOTD_SHOW_UPDATES="true"
# Preferred AUR helper: paru, yay, or auto (auto-detect)
AUR_HELPER="auto"
# ============================================================================ # ============================================================================
# Derived URLs (generally don't edit these) # Derived URLs (generally don't edit these)
# ============================================================================ # ============================================================================

View File

@@ -244,15 +244,26 @@ if _has_cmd kubectl; then
fi fi
# ============================================================================ # ============================================================================
# Dotfiles Functions (deferred loading) # Dotfiles Configuration
# ============================================================================ # ============================================================================
_dotfiles_dir="$HOME/.dotfiles" _dotfiles_dir="$HOME/.dotfiles"
# Load dotfiles.conf first (sets DOTFILES_DIR and other vars)
if [[ -f "$_dotfiles_dir/dotfiles.conf" ]]; then
source "$_dotfiles_dir/dotfiles.conf"
else
DOTFILES_DIR="$HOME/.dotfiles"
DOTFILES_BRANCH="main"
fi
# Source shared colors library
[[ -f "$_dotfiles_dir/zsh/lib/colors.zsh" ]] && source "$_dotfiles_dir/zsh/lib/colors.zsh"
# Source dotfiles aliases # Source dotfiles aliases
[[ -f "$_dotfiles_dir/zsh/aliases.zsh" ]] && source "$_dotfiles_dir/zsh/aliases.zsh" [[ -f "$_dotfiles_dir/zsh/aliases.zsh" ]] && source "$_dotfiles_dir/zsh/aliases.zsh"
# These are loaded immediately (small files, needed for keybindings) # Load command-palette immediately (needed for keybindings)
[[ -f "$_dotfiles_dir/zsh/functions/command-palette.zsh" ]] && \ [[ -f "$_dotfiles_dir/zsh/functions/command-palette.zsh" ]] && \
source "$_dotfiles_dir/zsh/functions/command-palette.zsh" source "$_dotfiles_dir/zsh/functions/command-palette.zsh"
@@ -267,34 +278,24 @@ _deferred_load() {
# Setup FZF # Setup FZF
_has_cmd fzf && _setup_fzf _has_cmd fzf && _setup_fzf
# Source optional function files # -----------------------------------------------------------------------
[[ -f "$_dotfiles_dir/zsh/functions/snapper.zsh" ]] && \ # Load all function files from functions directory
source "$_dotfiles_dir/zsh/functions/snapper.zsh" # Excludes command-palette.zsh (already loaded) and motd.zsh (loaded separately)
[[ -f "$_dotfiles_dir/zsh/functions/smart-suggest.zsh" ]] && \ # -----------------------------------------------------------------------
source "$_dotfiles_dir/zsh/functions/smart-suggest.zsh" local func_dir="$_dotfiles_dir/zsh/functions"
[[ -f "$_dotfiles_dir/zsh/functions/password-manager.zsh" ]] && \ if [[ -d "$func_dir" ]]; then
source "$_dotfiles_dir/zsh/functions/password-manager.zsh" for func_file in "$func_dir"/*.zsh; do
[[ -f "$_dotfiles_dir/zsh/functions/tmux-workspaces.zsh" ]] && \ [[ -f "$func_file" ]] || continue
source "$_dotfiles_dir/zsh/functions/tmux-workspaces.zsh"
[[ -f "$_dotfiles_dir/zsh/functions/python-templates.zsh" ]] && \
source "$_dotfiles_dir/zsh/functions/python-templates.zsh"
# Load vault secrets # Skip files that are loaded elsewhere
#local vault_script="$_dotfiles_dir/bin/dotfiles-vault.sh" case "${func_file:t}" in
#if [[ -f "$_dotfiles_dir/vault/secrets.enc" ]] && [[ -x "$vault_script" ]]; then command-palette.zsh) continue ;; # Loaded early for keybindings
# eval "$("$vault_script" shell 2>/dev/null)" || true motd.zsh) continue ;; # Loaded after prompt
#fi esac
# Load dotfiles.conf env variables. source "$func_file"
DOTFILES_CONF="$HOME/.dotfiles/dotfiles.conf" done
if [[ -f "$DOTFILES_CONF" ]]; then
source $DOTFILES_CONF
else
DOTFILES_DIR="$HOME/.dotfiles"
DOTFILES_BRANCH="main"
fi fi
} }
# ============================================================================ # ============================================================================
@@ -304,18 +305,20 @@ _deferred_load() {
_background_tasks() { _background_tasks() {
# Check for dotfiles updates # Check for dotfiles updates
if [[ "${DOTFILES_AUTO_SYNC_CHECK:-true}" == "true" ]]; then if [[ "${DOTFILES_AUTO_SYNC_CHECK:-true}" == "true" ]]; then
# Use full path to avoid command_not_found issues
$_dotfiles_dir/bin/dotfiles-sync.sh status -s 2> /dev/null $_dotfiles_dir/bin/dotfiles-sync.sh status -s 2> /dev/null
#[[ -x "$sync_script" ]] && "$sync_script" --auto 2>/dev/null &!
fi fi
_df_check_sys_updates _df_check_sys_updates
} }
_df_check_sys_updates() { _df_check_sys_updates() {
# Check number of available updates and export. # Check number of available updates and export
export UPDATE_PKG_COUNT=$(checkupdates | wc -l) if _has_cmd checkupdates; then
export UPDATE_PKG_COUNT=$(checkupdates 2>/dev/null | wc -l)
else
export UPDATE_PKG_COUNT=0
fi
} }
# ============================================================================ # ============================================================================
# Initialization Strategy # Initialization Strategy
# ============================================================================ # ============================================================================
@@ -344,6 +347,7 @@ else
case "${MOTD_STYLE:-compact}" in case "${MOTD_STYLE:-compact}" in
compact) show_motd ;; compact) show_motd ;;
mini) show_motd_mini ;; mini) show_motd_mini ;;
full) show_motd_full ;;
esac esac
fi fi

View File

@@ -0,0 +1,441 @@
# ============================================================================
# Btrfs Helpers for Arch/CachyOS
# ============================================================================
# Quick commands for btrfs filesystem management
# CachyOS defaults to btrfs, so these are highly useful
#
# Commands:
# btrfs-usage - Show filesystem usage
# btrfs-subs - List subvolumes
# btrfs-balance - Start balance operation
# btrfs-scrub - Start/check scrub
# btrfs-defrag - Defragment file or directory
# btrfs-compress - Show compression stats
# btrfs-info - Full filesystem info
# btrfs-health - Quick health check
# ============================================================================
# Source shared colors (with fallback)
source "${0:A:h}/../lib/colors.zsh" 2>/dev/null || \
source "$HOME/.dotfiles/zsh/lib/colors.zsh" 2>/dev/null || {
typeset -g DF_GREEN=$'\033[0;32m' DF_YELLOW=$'\033[1;33m'
typeset -g DF_RED=$'\033[0;31m' DF_BLUE=$'\033[0;34m'
typeset -g DF_CYAN=$'\033[0;36m' DF_NC=$'\033[0m'
}
# ============================================================================
# Configuration
# ============================================================================
typeset -g BTRFS_DEFAULT_MOUNT="${BTRFS_DEFAULT_MOUNT:-/}"
# ============================================================================
# Detection
# ============================================================================
_btrfs_check() {
if ! command -v btrfs &>/dev/null; then
echo -e "${DF_RED}${DF_NC} btrfs-progs not installed"
echo "Install: sudo pacman -S btrfs-progs"
return 1
fi
# Check if root is btrfs
local fstype=$(df -T / | awk 'NR==2 {print $2}')
if [[ "$fstype" != "btrfs" ]]; then
echo -e "${DF_YELLOW}${DF_NC} Root filesystem is not btrfs (detected: $fstype)"
return 1
fi
return 0
}
# ============================================================================
# Core Commands
# ============================================================================
# Show filesystem usage
btrfs-usage() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Filesystem Usage: ${mount} "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
sudo btrfs filesystem usage "$mount" -h
}
# List all subvolumes
btrfs-subs() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Subvolumes "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
echo -e "${DF_CYAN}Subvolume List:${DF_NC}"
sudo btrfs subvolume list "$mount" | while read -r line; do
local path=$(echo "$line" | awk '{print $NF}')
local id=$(echo "$line" | awk '{print $2}')
echo -e " ${DF_GREEN}${DF_NC} [$id] $path"
done
echo ""
echo -e "${DF_CYAN}Default Subvolume:${DF_NC}"
sudo btrfs subvolume get-default "$mount"
}
# Start balance operation
btrfs-balance() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
local usage="${2:-50}" # Default: rebalance chunks with <50% usage
echo -e "${DF_BLUE}==>${DF_NC} Starting btrfs balance on ${mount}"
echo -e "${DF_YELLOW}${DF_NC} This may take a while and use significant I/O"
echo ""
read -q "REPLY?Continue? [y/N]: "; echo
[[ ! "$REPLY" =~ ^[Yy]$ ]] && return 0
echo ""
echo -e "${DF_BLUE}==>${DF_NC} Balancing data chunks with <${usage}% usage..."
sudo btrfs balance start -dusage="$usage" -musage="$usage" "$mount" -v
if [[ $? -eq 0 ]]; then
echo -e "${DF_GREEN}${DF_NC} Balance completed"
else
echo -e "${DF_YELLOW}${DF_NC} Balance finished (may have been interrupted or had no work)"
fi
}
# Check balance status
btrfs-balance-status() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
sudo btrfs balance status "$mount"
}
# Cancel running balance
btrfs-balance-cancel() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}==>${DF_NC} Cancelling balance on ${mount}..."
sudo btrfs balance cancel "$mount"
}
# Start scrub operation
btrfs-scrub() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Scrub "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
# Check if scrub is already running
local status=$(sudo btrfs scrub status "$mount" 2>/dev/null)
if echo "$status" | grep -q "running"; then
echo -e "${DF_CYAN}Scrub Status (running):${DF_NC}"
echo "$status" | sed 's/^/ /'
return 0
fi
echo -e "${DF_YELLOW}${DF_NC} Scrub verifies data integrity and may take hours"
read -q "REPLY?Start scrub? [y/N]: "; echo
[[ ! "$REPLY" =~ ^[Yy]$ ]] && return 0
echo ""
echo -e "${DF_BLUE}==>${DF_NC} Starting scrub..."
sudo btrfs scrub start "$mount"
echo ""
echo -e "${DF_CYAN}Scrub Status:${DF_NC}"
sudo btrfs scrub status "$mount"
echo ""
echo -e "${DF_CYAN}Monitor with:${DF_NC} btrfs-scrub-status"
}
# Show scrub status
btrfs-scrub-status() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
sudo btrfs scrub status "$mount"
}
# Cancel scrub
btrfs-scrub-cancel() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}==>${DF_NC} Cancelling scrub on ${mount}..."
sudo btrfs scrub cancel "$mount"
}
# Defragment file or directory
btrfs-defrag() {
_btrfs_check || return 1
local target="${1:-.}"
if [[ ! -e "$target" ]]; then
echo -e "${DF_RED}${DF_NC} Target not found: $target"
return 1
fi
echo -e "${DF_BLUE}==>${DF_NC} Defragmenting: $target"
if [[ -d "$target" ]]; then
echo -e "${DF_YELLOW}${DF_NC} Recursive defrag on directory"
read -q "REPLY?Continue? [y/N]: "; echo
[[ ! "$REPLY" =~ ^[Yy]$ ]] && return 0
sudo btrfs filesystem defragment -r -v "$target"
else
sudo btrfs filesystem defragment -v "$target"
fi
echo -e "${DF_GREEN}${DF_NC} Defragmentation complete"
}
# Show compression stats (requires compsize)
btrfs-compress() {
_btrfs_check || return 1
local target="${1:-$BTRFS_DEFAULT_MOUNT}"
if ! command -v compsize &>/dev/null; then
echo -e "${DF_YELLOW}${DF_NC} compsize not installed"
echo "Install: sudo pacman -S compsize"
return 1
fi
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Compression Statistics "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
sudo compsize "$target"
}
# ============================================================================
# Information Commands
# ============================================================================
# Full filesystem info
btrfs-info() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Filesystem Information "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}Filesystem Show:${DF_NC}"
sudo btrfs filesystem show "$mount"
echo -e "\n${DF_CYAN}Filesystem df:${DF_NC}"
sudo btrfs filesystem df "$mount"
echo -e "\n${DF_CYAN}Device Stats:${DF_NC}"
sudo btrfs device stats "$mount"
echo ""
}
# Quick health check
btrfs-health() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Health Check "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
local issues=0
# Check device stats for errors
echo -e "${DF_CYAN}Device Errors:${DF_NC}"
local stats=$(sudo btrfs device stats "$mount" 2>/dev/null)
local errors=$(echo "$stats" | grep -v " 0$" | grep -v "^$")
if [[ -z "$errors" ]]; then
echo -e " ${DF_GREEN}${DF_NC} No device errors detected"
else
echo -e " ${DF_RED}${DF_NC} Errors detected:"
echo "$errors" | sed 's/^/ /'
issues=$((issues + 1))
fi
# Check allocation
echo -e "\n${DF_CYAN}Space Allocation:${DF_NC}"
local usage=$(sudo btrfs filesystem usage "$mount" -b 2>/dev/null)
local used_pct=$(echo "$usage" | grep "Used:" | head -1 | awk '{print $2}' | tr -d '%')
if [[ -n "$used_pct" ]]; then
if (( used_pct >= 90 )); then
echo -e " ${DF_RED}${DF_NC} Filesystem ${used_pct}% full - critical!"
issues=$((issues + 1))
elif (( used_pct >= 80 )); then
echo -e " ${DF_YELLOW}${DF_NC} Filesystem ${used_pct}% full - consider cleanup"
else
echo -e " ${DF_GREEN}${DF_NC} Filesystem ${used_pct}% used"
fi
fi
# Check last scrub
echo -e "\n${DF_CYAN}Last Scrub:${DF_NC}"
local scrub_status=$(sudo btrfs scrub status "$mount" 2>/dev/null)
local scrub_date=$(echo "$scrub_status" | grep "Scrub started" | awk '{print $3, $4, $5}')
local scrub_errors=$(echo "$scrub_status" | grep "Error summary" | grep -v "no errors")
if [[ -n "$scrub_date" ]]; then
echo -e " Last scrub: $scrub_date"
if [[ -n "$scrub_errors" ]]; then
echo -e " ${DF_RED}${DF_NC} Scrub found errors"
echo "$scrub_errors" | sed 's/^/ /'
issues=$((issues + 1))
else
echo -e " ${DF_GREEN}${DF_NC} No errors in last scrub"
fi
else
echo -e " ${DF_YELLOW}${DF_NC} No scrub has been run (recommended monthly)"
fi
# Summary
echo ""
if (( issues == 0 )); then
echo -e "${DF_GREEN}${DF_NC} Btrfs filesystem appears healthy"
else
echo -e "${DF_RED}${DF_NC} Found $issues issue(s) - investigate above"
fi
echo ""
}
# ============================================================================
# Snapshot Helpers (complement snapper.zsh)
# ============================================================================
# Show snapshot space usage
btrfs-snap-usage() {
_btrfs_check || return 1
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Snapshot Space Usage "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
if [[ -d "/.snapshots" ]]; then
echo -e "${DF_CYAN}Snapshot Directory:${DF_NC}"
sudo du -sh /.snapshots 2>/dev/null || echo " Unable to calculate"
echo -e "\n${DF_CYAN}Individual Snapshots:${DF_NC}"
sudo du -sh /.snapshots/*/ 2>/dev/null | sort -h | tail -10 | sed 's/^/ /'
else
echo -e "${DF_YELLOW}${DF_NC} No /.snapshots directory found"
fi
echo ""
}
# ============================================================================
# Maintenance
# ============================================================================
# Full maintenance routine
btrfs-maintain() {
_btrfs_check || return 1
local mount="${1:-$BTRFS_DEFAULT_MOUNT}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Maintenance Routine "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo ""
echo "This will perform:"
echo " 1. Health check"
echo " 2. Balance (low usage chunks)"
echo " 3. Scrub (data integrity)"
echo ""
echo -e "${DF_YELLOW}${DF_NC} This may take several hours"
read -q "REPLY?Continue? [y/N]: "; echo
[[ ! "$REPLY" =~ ^[Yy]$ ]] && return 0
echo ""
echo -e "${DF_BLUE}==>${DF_NC} Step 1/3: Health Check"
btrfs-health "$mount"
echo -e "${DF_BLUE}==>${DF_NC} Step 2/3: Balance"
sudo btrfs balance start -dusage=50 -musage=50 "$mount"
echo ""
echo -e "${DF_BLUE}==>${DF_NC} Step 3/3: Scrub"
sudo btrfs scrub start -B "$mount" # -B runs in foreground
echo ""
echo -e "${DF_GREEN}${DF_NC} Maintenance complete"
btrfs-health "$mount"
}
# ============================================================================
# Aliases
# ============================================================================
alias btru='btrfs-usage'
alias btrs='btrfs-subs'
alias btrh='btrfs-health'
alias btri='btrfs-info'
alias btrc='btrfs-compress'
# ============================================================================
# Help
# ============================================================================
btrfs-help() {
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Btrfs Helper Commands "
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
cat << 'EOF'
Information:
btrfs-usage [mount] Filesystem usage summary
btrfs-subs [mount] List all subvolumes
btrfs-info [mount] Full filesystem information
btrfs-health [mount] Quick health check
btrfs-compress [path] Compression statistics (requires compsize)
Maintenance:
btrfs-balance [mount] Start balance operation
btrfs-balance-status Check balance progress
btrfs-balance-cancel Cancel running balance
btrfs-scrub [mount] Start scrub (integrity check)
btrfs-scrub-status Check scrub progress
btrfs-scrub-cancel Cancel running scrub
btrfs-defrag <path> Defragment file/directory
btrfs-maintain [mount] Full maintenance routine
Snapshots:
btrfs-snap-usage Show snapshot space usage
Aliases:
btru btrfs-usage
btrs btrfs-subs
btrh btrfs-health
btri btrfs-info
btrc btrfs-compress
Note: Most commands default to / if no mount point specified.
See also: snapper.zsh for snapshot management
EOF
}

View File

@@ -3,10 +3,12 @@
# MOTD (Message of the Day) - Dynamic System Info # MOTD (Message of the Day) - Dynamic System Info
# ============================================================================ # ============================================================================
# Displays system information on shell startup # Displays system information on shell startup
# Optimized for Arch/CachyOS with direct /proc access
# #
# Functions: # Functions:
# show_motd - Compact box format # show_motd - Compact box format
# show_motd_mini - Single line format # show_motd_mini - Single line format
# show_motd_full - Extended info
# ============================================================================ # ============================================================================
# Only run in interactive shells # Only run in interactive shells
@@ -18,7 +20,7 @@ source "$HOME/.dotfiles/zsh/lib/colors.zsh" 2>/dev/null || {
typeset -g DF_RESET=$'\033[0m' DF_BOLD=$'\033[1m' DF_DIM=$'\033[2m' typeset -g DF_RESET=$'\033[0m' DF_BOLD=$'\033[1m' DF_DIM=$'\033[2m'
typeset -g DF_BLUE=$'\033[38;5;39m' DF_CYAN=$'\033[38;5;51m' typeset -g DF_BLUE=$'\033[38;5;39m' DF_CYAN=$'\033[38;5;51m'
typeset -g DF_GREEN=$'\033[38;5;82m' DF_YELLOW=$'\033[38;5;220m' typeset -g DF_GREEN=$'\033[38;5;82m' DF_YELLOW=$'\033[38;5;220m'
typeset -g DF_GREY=$'\033[38;5;242m' DF_NC=$'\033[0m' typeset -g DF_RED=$'\033[38;5;196m' DF_GREY=$'\033[38;5;242m' DF_NC=$'\033[0m'
} }
# ============================================================================ # ============================================================================
@@ -28,32 +30,98 @@ source "$HOME/.dotfiles/zsh/lib/colors.zsh" 2>/dev/null || {
typeset -g _M_WIDTH=66 typeset -g _M_WIDTH=66
# ============================================================================ # ============================================================================
# Info Gathering # Optimized Info Gathering (using /proc directly - faster than spawning processes)
# ============================================================================ # ============================================================================
# Uptime from /proc (no subprocess)
_motd_uptime() { _motd_uptime() {
local up=$(uptime 2>/dev/null) local uptime_seconds=$(cut -d. -f1 /proc/uptime 2>/dev/null)
if [[ "$up" =~ "up "([^,]+) ]]; then [[ -z "$uptime_seconds" ]] && { echo "?"; return; }
echo "${match[1]}" | sed 's/^ *//'
local days=$((uptime_seconds / 86400))
local hours=$(((uptime_seconds % 86400) / 3600))
local mins=$(((uptime_seconds % 3600) / 60))
if (( days > 0 )); then
echo "${days}d ${hours}h"
elif (( hours > 0 )); then
echo "${hours}h ${mins}m"
else else
echo "?" echo "${mins}m"
fi fi
} }
# Load from /proc (no subprocess)
_motd_load() { _motd_load() {
if [[ -f /proc/loadavg ]]; then cut -d' ' -f1 /proc/loadavg 2>/dev/null || echo "?"
awk '{print $1}' /proc/loadavg }
# Memory from /proc (single awk call)
_motd_mem() {
awk '/MemTotal/ {total=$2} /MemAvailable/ {avail=$2} END {
if (total > 0) {
used=(total-avail)/1024/1024
total_gb=total/1024/1024
printf "%.1fG/%.0fG", used, total_gb
} else {
print "N/A"
}
}' /proc/meminfo 2>/dev/null || echo "N/A"
}
# Disk usage (single df call)
_motd_disk() {
df -h / 2>/dev/null | awk 'NR==2 {print $3 "/" $2}' || echo "N/A"
}
# CPU governor (Arch-specific, direct file read)
_motd_governor() {
local gov=$(cat /sys/devices/system/cpu/cpu0/cpufreq/scaling_governor 2>/dev/null)
[[ -n "$gov" ]] && echo "$gov"
}
# Kernel version (simplified)
_motd_kernel() {
local kernel=$(uname -r)
# Strip architecture suffix for cleaner display
echo "${kernel%%-*}"
}
# CachyOS scheduler detection
_motd_scheduler() {
if grep -q "cachyos" /proc/version 2>/dev/null; then
if grep -q "bore" /proc/version 2>/dev/null; then
echo "BORE"
elif grep -q "eevdf" /proc/version 2>/dev/null; then
echo "EEVDF"
else else
uptime | awk -F'load average:' '{print $2}' | awk -F, '{print $1}' | xargs echo "CachyOS"
fi
fi fi
} }
_motd_mem() { # Failed systemd services count (cached)
free -h 2>/dev/null | awk '/^Mem:/ {print $3 "/" $2}' || echo "N/A" _motd_failed_services() {
# Use cache to avoid slow systemctl calls on every prompt
local cache_file="/tmp/.motd-failed-${UID}"
local cache_age=300 # 5 minutes
if [[ -f "$cache_file" ]]; then
local file_age=$(($(date +%s) - $(stat -c %Y "$cache_file" 2>/dev/null || echo 0)))
if (( file_age < cache_age )); then
cat "$cache_file"
return
fi
fi
local count=$(systemctl --failed --no-pager --no-legend 2>/dev/null | wc -l)
echo "$count" > "$cache_file" 2>/dev/null
echo "$count"
} }
_motd_disk() { # Package updates (from environment, set by aliases.zsh)
df -h / 2>/dev/null | awk 'NR==2 {print $3 "/" $4}' || echo "N/A" _motd_updates() {
echo "${UPDATE_PKG_COUNT:-0}"
} }
# ============================================================================ # ============================================================================
@@ -62,27 +130,13 @@ _motd_disk() {
_motd_line() { _motd_line() {
local char="$1" local char="$1"
local i
local line="" local line=""
for ((i=0; i<_M_WIDTH; i++)); do for ((i=0; i<_M_WIDTH; i++)); do line+="$char"; done
line+="$char"
done
echo "$line" echo "$line"
} }
_motd_pad() {
local str="$1"
local width="$2"
local len=${#str}
if (( len >= width )); then
echo "${str:0:$width}"
else
printf "%-${width}s" "$str"
fi
}
# ============================================================================ # ============================================================================
# Main Display Function # Main Display Function (Compact)
# ============================================================================ # ============================================================================
show_motd() { show_motd() {
@@ -95,7 +149,6 @@ show_motd() {
local load=$(_motd_load) local load=$(_motd_load)
local mem=$(_motd_mem) local mem=$(_motd_mem)
local disk=$(_motd_disk) local disk=$(_motd_disk)
local local_ip=$(hostname -i 2>/dev/null | awk -F" " '{print $1}' || echo "N/A")
local hline=$(_motd_line '═') local hline=$(_motd_line '═')
local inner=$((_M_WIDTH - 2)) local inner=$((_M_WIDTH - 2))
@@ -106,22 +159,38 @@ show_motd() {
# Header: hostname + datetime # Header: hostname + datetime
local h_left="${hostname}" local h_left="${hostname}"
local h_center="${local_ip}"
local h_right="${datetime}" local h_right="${datetime}"
local h_pad=$(((inner - ${#h_left} - ${#h_center} - ${#h_right}) / 2 )) local h_pad=$(((inner - ${#h_left} - ${#h_right}) / 2))
local h_spaces="" local h_spaces=""
for ((i=0; i<h_pad; i++)); do h_spaces+=" "; done for ((i=0; i<h_pad; i++)); do h_spaces+=" "; done
echo "${DF_GREY}${DF_NC} ${DF_BOLD}${DF_BLUE}${h_left}${DF_NC}${h_spaces}${DF_YELLOW}${h_center}${h_spaces}${DF_NC}${DF_BOLD}${h_right}${DF_NC}${DF_GREY} ${DF_NC}" echo "${DF_GREY}${DF_NC} ${DF_BOLD}${DF_BLUE}${h_left}${DF_NC}${h_spaces}${h_spaces}${DF_NC}${DF_BOLD}${h_right}${DF_NC} ${DF_GREY}${DF_NC}"
# Separator # Separator
echo "${DF_GREY}${hline}${DF_NC}" echo "${DF_GREY}${hline}${DF_NC}"
# Stats line # Stats line
local s1="${DF_YELLOW} up:${DF_NC}${uptime}" local s1="${DF_YELLOW}${DF_NC}${uptime}"
local s2="${DF_CYAN} load:${DF_NC}${load}" local s2="${DF_CYAN}${DF_NC}${load}"
local s3="${DF_GREEN} mem:${DF_NC}${mem}" local s3="${DF_GREEN}${DF_NC}${mem}"
local s4="${DF_BLUE}${DF_NC}${disk}" local s4="${DF_BLUE}${DF_NC}${disk}"
echo "${DF_GREY}${DF_DIM}${DF_NC}${s1}${DF_GREY}${DF_DIM}〙⎯〘${s2}${DF_GREY}${DF_DIM}〙⎯〘${s3}${DF_GREY}${DF_DIM}〙⎯〘${s4}${DF_GREY}${DF_DIM}${DF_NC}" echo " ${s1} ${s2} ${s3} ${s4}"
# Alerts line (if any issues)
local alerts=""
# Check for failed services
local failed=$(_motd_failed_services)
if (( failed > 0 )); then
alerts+="${DF_RED}${failed} failed service(s)${DF_NC} "
fi
# Check for updates
local updates=$(_motd_updates)
if (( updates > 0 )); then
alerts+="${DF_YELLOW}${updates} update(s)${DF_NC}"
fi
[[ -n "$alerts" ]] && echo " $alerts"
echo "" echo ""
} }
@@ -137,8 +206,64 @@ show_motd_mini() {
local hostname="${HOST:-$(hostname -s 2>/dev/null)}" local hostname="${HOST:-$(hostname -s 2>/dev/null)}"
local uptime=$(_motd_uptime) local uptime=$(_motd_uptime)
local mem=$(_motd_mem) local mem=$(_motd_mem)
local failed=$(_motd_failed_services)
echo "${DF_DIM}──${DF_NC} ${DF_BOLD}${hostname}${DF_NC} ${DF_DIM}${DF_NC} up:${uptime} ${DF_DIM}${DF_NC} mem:${mem} ${DF_DIM}──${DF_NC}" local alert=""
(( failed > 0 )) && alert=" ${DF_RED}[${failed} failed]${DF_NC}"
echo "${DF_DIM}──${DF_NC} ${DF_BOLD}${hostname}${DF_NC} ${DF_DIM}${DF_NC} up:${uptime} ${DF_DIM}${DF_NC} mem:${mem}${alert} ${DF_DIM}──${DF_NC}"
}
# ============================================================================
# Full Format (Extended Info)
# ============================================================================
show_motd_full() {
[[ -n "$_MOTD_SHOWN" && "$1" != "--force" ]] && return 0
typeset -g _MOTD_SHOWN=1
local hostname="${HOST:-$(hostname -s 2>/dev/null)}"
local datetime=$(date '+%A, %B %d %Y %H:%M:%S')
local uptime=$(_motd_uptime)
local load=$(_motd_load)
local mem=$(_motd_mem)
local disk=$(_motd_disk)
local kernel=$(_motd_kernel)
local governor=$(_motd_governor)
local scheduler=$(_motd_scheduler)
local hline=$(_motd_line '═')
echo ""
echo "${DF_GREY}${hline}${DF_NC}"
echo "${DF_GREY}${DF_NC} ${DF_BOLD}${DF_BLUE}${hostname}${DF_NC}"
echo "${DF_GREY}${DF_NC} ${DF_DIM}${datetime}${DF_NC}"
echo "${DF_GREY}$(_motd_line '─')${DF_NC}"
# System info
echo "${DF_GREY}${DF_NC} ${DF_CYAN}Kernel:${DF_NC} ${kernel}"
[[ -n "$scheduler" ]] && echo "${DF_GREY}${DF_NC} ${DF_CYAN}Scheduler:${DF_NC} ${scheduler}"
[[ -n "$governor" ]] && echo "${DF_GREY}${DF_NC} ${DF_CYAN}Governor:${DF_NC} ${governor}"
echo "${DF_GREY}$(_motd_line '─')${DF_NC}"
# Resources
echo "${DF_GREY}${DF_NC} ${DF_YELLOW}▲ Uptime:${DF_NC} ${uptime}"
echo "${DF_GREY}${DF_NC} ${DF_CYAN}◆ Load:${DF_NC} ${load}"
echo "${DF_GREY}${DF_NC} ${DF_GREEN}◇ Memory:${DF_NC} ${mem}"
echo "${DF_GREY}${DF_NC} ${DF_BLUE}⊡ Disk:${DF_NC} ${disk}"
# Alerts section
local failed=$(_motd_failed_services)
local updates=$(_motd_updates)
if (( failed > 0 || updates > 0 )); then
echo "${DF_GREY}$(_motd_line '─')${DF_NC}"
(( failed > 0 )) && echo "${DF_GREY}${DF_NC} ${DF_RED}${failed} failed systemd service(s)${DF_NC}"
(( updates > 0 )) && echo "${DF_GREY}${DF_NC} ${DF_YELLOW}${updates} package update(s) available${DF_NC}"
fi
echo "${DF_GREY}${hline}${DF_NC}"
echo ""
} }
# ============================================================================ # ============================================================================
@@ -147,3 +272,29 @@ show_motd_mini() {
alias motd='show_motd --force' alias motd='show_motd --force'
alias motd-mini='show_motd_mini --force' alias motd-mini='show_motd_mini --force'
alias motd-full='show_motd_full --force'
# ============================================================================
# Quick System Overview (callable anytime)
# ============================================================================
sysbrief() {
echo -e "${DF_CYAN}Uptime:${DF_NC} $(_motd_uptime)"
echo -e "${DF_CYAN}Load:${DF_NC} $(_motd_load)"
echo -e "${DF_CYAN}Memory:${DF_NC} $(_motd_mem)"
echo -e "${DF_CYAN}Disk:${DF_NC} $(_motd_disk)"
local kernel=$(_motd_kernel)
echo -e "${DF_CYAN}Kernel:${DF_NC} ${kernel}"
local governor=$(_motd_governor)
[[ -n "$governor" ]] && echo -e "${DF_CYAN}Governor:${DF_NC} ${governor}"
local scheduler=$(_motd_scheduler)
[[ -n "$scheduler" ]] && echo -e "${DF_CYAN}Scheduler:${DF_NC} ${scheduler}"
local failed=$(_motd_failed_services)
if (( failed > 0 )); then
echo -e "${DF_RED}Failed:${DF_NC} ${failed} service(s)"
fi
}

View File

@@ -0,0 +1,348 @@
# ============================================================================
# Systemd Integration for Arch/CachyOS
# ============================================================================
# Quick shortcuts and helpers for systemd service management
#
# Commands:
# sc <args> - sudo systemctl
# scu <args> - systemctl --user
# scr <service> - restart and show status
# sce <service> - enable and start
# scd <service> - disable and stop
# sclog <service> - follow journal logs
# sc-failed - show failed services
# sc-timers - show active timers
# sc-recent - recently started services
# sc-boot - boot time analysis
# ============================================================================
# Source shared colors (with fallback)
source "${0:A:h}/../lib/colors.zsh" 2>/dev/null || \
source "$HOME/.dotfiles/zsh/lib/colors.zsh" 2>/dev/null || {
typeset -g DF_GREEN=$'\033[0;32m' DF_YELLOW=$'\033[1;33m'
typeset -g DF_RED=$'\033[0;31m' DF_BLUE=$'\033[0;34m'
typeset -g DF_CYAN=$'\033[0;36m' DF_NC=$'\033[0m'
}
# ============================================================================
# Core Systemctl Shortcuts
# ============================================================================
# System-level systemctl (with sudo)
sc() {
sudo systemctl "$@"
}
# User-level systemctl
scu() {
systemctl --user "$@"
}
# Restart service and show status
scr() {
local service="$1"
[[ -z "$service" ]] && { echo "Usage: scr <service>"; return 1; }
echo -e "${DF_BLUE}==>${DF_NC} Restarting ${service}..."
if sudo systemctl restart "$service"; then
echo -e "${DF_GREEN}${DF_NC} Restarted successfully"
echo ""
sudo systemctl status "$service" --no-pager -l
else
echo -e "${DF_RED}${DF_NC} Failed to restart ${service}"
return 1
fi
}
# Enable and start service
sce() {
local service="$1"
[[ -z "$service" ]] && { echo "Usage: sce <service>"; return 1; }
echo -e "${DF_BLUE}==>${DF_NC} Enabling and starting ${service}..."
if sudo systemctl enable --now "$service"; then
echo -e "${DF_GREEN}${DF_NC} ${service} enabled and started"
sudo systemctl status "$service" --no-pager -l | head -15
else
echo -e "${DF_RED}${DF_NC} Failed to enable ${service}"
return 1
fi
}
# Disable and stop service
scd() {
local service="$1"
[[ -z "$service" ]] && { echo "Usage: scd <service>"; return 1; }
echo -e "${DF_BLUE}==>${DF_NC} Disabling and stopping ${service}..."
if sudo systemctl disable --now "$service"; then
echo -e "${DF_GREEN}${DF_NC} ${service} disabled and stopped"
else
echo -e "${DF_RED}${DF_NC} Failed to disable ${service}"
return 1
fi
}
# Follow journal logs for a service
sclog() {
local service="$1"
local lines="${2:-50}"
[[ -z "$service" ]] && { echo "Usage: sclog <service> [lines]"; return 1; }
echo -e "${DF_BLUE}==>${DF_NC} Following logs for ${service} (Ctrl+C to exit)..."
sudo journalctl -xeu "$service" -f -n "$lines"
}
# Show recent logs for a service (without follow)
sclogs() {
local service="$1"
local lines="${2:-50}"
[[ -z "$service" ]] && { echo "Usage: sclog <service> [lines]"; return 1; }
sudo journalctl -xeu "$service" -n "$lines" --no-pager
}
# ============================================================================
# Service Status Commands
# ============================================================================
# Show failed services (system and user)
sc-failed() {
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Failed Services ${DF_BLUE}${DF_NC}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}System Services:${DF_NC}"
local sys_failed=$(systemctl --failed --no-pager --no-legend 2>/dev/null)
if [[ -z "$sys_failed" ]]; then
echo -e " ${DF_GREEN}${DF_NC} No failed system services"
else
echo "$sys_failed" | sed 's/^/ /'
fi
echo -e "\n${DF_CYAN}User Services:${DF_NC}"
local user_failed=$(systemctl --user --failed --no-pager --no-legend 2>/dev/null)
if [[ -z "$user_failed" ]]; then
echo -e " ${DF_GREEN}${DF_NC} No failed user services"
else
echo "$user_failed" | sed 's/^/ /'
fi
echo ""
}
# Show active timers
sc-timers() {
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Active Timers ${DF_BLUE}${DF_NC}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}System Timers:${DF_NC}"
systemctl list-timers --no-pager | head -20
echo -e "\n${DF_CYAN}User Timers:${DF_NC}"
systemctl --user list-timers --no-pager 2>/dev/null | head -10
echo ""
}
# Show recently started/stopped services
sc-recent() {
local count="${1:-15}"
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Recent Service Activity ${DF_BLUE}${DF_NC}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}Recently Started:${DF_NC}"
systemctl list-units --type=service --state=running --no-pager --no-legend | \
head -"$count" | awk '{print " " $1}'
echo -e "\n${DF_CYAN}Recent Journal (services):${DF_NC}"
journalctl -p 3 -xb --no-pager | tail -"$count" | sed 's/^/ /'
echo ""
}
# Boot time analysis
sc-boot() {
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Boot Time Analysis ${DF_BLUE}${DF_NC}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}Boot Summary:${DF_NC}"
systemd-analyze
echo -e "\n${DF_CYAN}Slowest Services (top 10):${DF_NC}"
systemd-analyze blame --no-pager | head -10 | sed 's/^/ /'
echo -e "\n${DF_CYAN}Critical Chain:${DF_NC}"
systemd-analyze critical-chain --no-pager 2>/dev/null | head -15 | sed 's/^/ /'
echo ""
}
# ============================================================================
# Service Search and Info
# ============================================================================
# Search for services by name
sc-search() {
local query="$1"
[[ -z "$query" ]] && { echo "Usage: sc-search <query>"; return 1; }
echo -e "${DF_BLUE}==>${DF_NC} Searching for services matching: ${query}"
echo ""
systemctl list-unit-files --type=service --no-pager | grep -i "$query"
}
# Show detailed service info
sc-info() {
local service="$1"
[[ -z "$service" ]] && { echo "Usage: sc-info <service>"; return 1; }
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Service Info: ${service}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
echo -e "\n${DF_CYAN}Status:${DF_NC}"
systemctl status "$service" --no-pager -l 2>/dev/null || \
sudo systemctl status "$service" --no-pager -l
echo -e "\n${DF_CYAN}Unit File:${DF_NC}"
systemctl cat "$service" 2>/dev/null | head -30
echo ""
}
# ============================================================================
# Quick Status for MOTD Integration
# ============================================================================
# Get count of failed services (for MOTD/prompt)
_systemd_failed_count() {
local count=$(systemctl --failed --no-pager --no-legend 2>/dev/null | wc -l)
echo "$count"
}
# Check if a service is active (for scripts)
_systemd_is_active() {
local service="$1"
systemctl is-active --quiet "$service" 2>/dev/null
}
# Check if a service is enabled (for scripts)
_systemd_is_enabled() {
local service="$1"
systemctl is-enabled --quiet "$service" 2>/dev/null
}
# ============================================================================
# Interactive Service Management (requires fzf)
# ============================================================================
if command -v fzf &>/dev/null; then
# Interactive service selector
scf() {
local service=$(systemctl list-units --type=service --no-pager --no-legend | \
awk '{print $1, $2, $3, $4}' | \
fzf --height=50% --layout=reverse --border=rounded \
--prompt='Service > ' \
--preview='systemctl status {1} --no-pager' \
--preview-window=right:50%:wrap | \
awk '{print $1}')
if [[ -n "$service" ]]; then
echo -e "${DF_BLUE}Selected:${DF_NC} $service"
echo ""
echo "Actions: [s]tatus [r]estart [o]stop [l]ogs [e]nable [d]isable [q]uit"
read -k 1 "action?Action: "
echo ""
case "$action" in
s) sudo systemctl status "$service" --no-pager -l ;;
r) scr "$service" ;;
o) sudo systemctl stop "$service" ;;
l) sclog "$service" ;;
e) sce "$service" ;;
d) scd "$service" ;;
q) return 0 ;;
*) echo "Unknown action" ;;
esac
fi
}
# Interactive log viewer
sclogf() {
local service=$(systemctl list-units --type=service --no-pager --no-legend | \
awk '{print $1}' | \
fzf --height=40% --layout=reverse --prompt='Service logs > ')
[[ -n "$service" ]] && sclog "$service"
}
fi
# ============================================================================
# Aliases
# ============================================================================
alias scs='sc status'
alias scstart='sc start'
alias scstop='sc stop'
alias screload='sc daemon-reload'
alias scmask='sc mask'
alias scunmask='sc unmask'
# Journal shortcuts
alias jctl='journalctl'
alias jctlf='journalctl -f'
alias jctlb='journalctl -b'
alias jctlerr='journalctl -p err -b'
# ============================================================================
# Help
# ============================================================================
sc-help() {
echo -e "${DF_BLUE}╔════════════════════════════════════════════════════════════╗${DF_NC}"
echo -e "${DF_BLUE}${DF_NC} Systemd Helper Commands ${DF_BLUE}${DF_NC}"
echo -e "${DF_BLUE}╚════════════════════════════════════════════════════════════╝${DF_NC}"
cat << 'EOF'
Core Commands:
sc <args> sudo systemctl <args>
scu <args> systemctl --user <args>
scr <service> Restart and show status
sce <service> Enable and start (--now)
scd <service> Disable and stop (--now)
sclog <service> Follow journal logs (-f)
scogs <service> Show recent logs (no follow)
Status Commands:
sc-failed Show failed services
sc-timers Show active timers
sc-recent Recently started services
sc-boot Boot time analysis
sc-search <query> Search services by name
sc-info <service> Detailed service info
Interactive (requires fzf):
scf Interactive service manager
sclogf Interactive log viewer
Aliases:
scs sc status
scstart sc start
scstop sc stop
screload sc daemon-reload
Journal:
jctl journalctl
jctlf journalctl -f
jctlb journalctl -b (current boot)
jctlerr journalctl -p err -b
EOF
}