Dotfiles update 2025-12-25 16:52
This commit is contained in:
@@ -309,18 +309,6 @@ alias hist="history"
|
|||||||
alias cls="clear"
|
alias cls="clear"
|
||||||
alias q="exit"
|
alias q="exit"
|
||||||
|
|
||||||
# System upgrade with snapshot (Arch/CachyOS)
|
|
||||||
sys-update() {
|
|
||||||
local update_date=$(date +"%Y-%m-%d %H:%M")
|
|
||||||
if command -v snapper &>/dev/null; then
|
|
||||||
sudo snapper -c root create --description "System Update ${update_date}" --command "sudo pacman -Syu"
|
|
||||||
else
|
|
||||||
sudo pacman -Syu
|
|
||||||
fi
|
|
||||||
# Update package count for prompt if available
|
|
||||||
command -v checkupdates &>/dev/null && export UPDATE_PKG_COUNT=$(checkupdates 2>/dev/null | wc -l)
|
|
||||||
}
|
|
||||||
|
|
||||||
# Markdown viewer with glow
|
# Markdown viewer with glow
|
||||||
if command -v glow &>/dev/null; then
|
if command -v glow &>/dev/null; then
|
||||||
alias glow='glow -p'
|
alias glow='glow -p'
|
||||||
|
|||||||
@@ -5,6 +5,10 @@
|
|||||||
source "${0:A:h}/../lib/utils.zsh" 2>/dev/null || \
|
source "${0:A:h}/../lib/utils.zsh" 2>/dev/null || \
|
||||||
source "$HOME/.dotfiles/zsh/lib/utils.zsh" 2>/dev/null
|
source "$HOME/.dotfiles/zsh/lib/utils.zsh" 2>/dev/null
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# Core Snapshot Functions
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
snap-create() {
|
snap-create() {
|
||||||
local desc="$*"
|
local desc="$*"
|
||||||
local limine="/boot/limine.conf"
|
local limine="/boot/limine.conf"
|
||||||
@@ -20,7 +24,7 @@ snap-create() {
|
|||||||
[[ ! -f "$limine" ]] && { df_print_error "Limine not found: $limine"; return 1; }
|
[[ ! -f "$limine" ]] && { df_print_error "Limine not found: $limine"; return 1; }
|
||||||
|
|
||||||
df_print_step "Checking limine.conf before snapshot"
|
df_print_step "Checking limine.conf before snapshot"
|
||||||
local before=$(sudo grep -cP "^\\s*///\\d+\\s*│" "$limine" || echo "0")
|
local before=$(sudo grep -cP "^\\s*///\\d+\\s*│" "$limine" 2>/dev/null || echo "0")
|
||||||
df_print_success "Before: $before entries"
|
df_print_success "Before: $before entries"
|
||||||
|
|
||||||
df_print_step "Creating snapshot: \"$desc\""
|
df_print_step "Creating snapshot: \"$desc\""
|
||||||
@@ -33,20 +37,22 @@ snap-create() {
|
|||||||
sleep 2
|
sleep 2
|
||||||
|
|
||||||
df_print_step "Validating"
|
df_print_step "Validating"
|
||||||
local after=$(sudo grep -cP "^\\s*///\\d+\\s*│" "$limine" || echo "0")
|
local after=$(sudo grep -cP "^\\s*///\\d+\\s*│" "$limine" 2>/dev/null || echo "0")
|
||||||
|
|
||||||
if sudo grep -qP "^\\s*///$num\\s*│" "$limine"; then
|
if sudo grep -qP "^\\s*///$num\\s*│" "$limine" 2>/dev/null; then
|
||||||
df_print_success "Snapshot #$num in limine.conf"
|
df_print_success "Snapshot #$num in limine.conf"
|
||||||
(( after > before )) && df_print_success "Added $((after - before)) entry"
|
(( after > before )) && df_print_success "Added $((after - before)) entry"
|
||||||
else
|
else
|
||||||
df_print_error "Snapshot #$num NOT in limine.conf"
|
df_print_warning "Snapshot #$num not yet in limine.conf (may sync later)"
|
||||||
return 1
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
df_print_section "Summary"
|
df_print_section "Summary"
|
||||||
df_print_indent "Number: #$num"
|
df_print_indent "Number: #$num"
|
||||||
df_print_indent "Description: $desc"
|
df_print_indent "Description: $desc"
|
||||||
|
|
||||||
|
# Return the snapshot number for use by other functions
|
||||||
|
echo "$num"
|
||||||
}
|
}
|
||||||
|
|
||||||
snap-list() {
|
snap-list() {
|
||||||
@@ -61,7 +67,7 @@ snap-show() {
|
|||||||
sudo snapper -c root list | grep "^\s*$1\s"
|
sudo snapper -c root list | grep "^\s*$1\s"
|
||||||
echo ""
|
echo ""
|
||||||
df_print_section "In limine.conf"
|
df_print_section "In limine.conf"
|
||||||
sudo grep -qP "^\\s*///$1\\s*│" /boot/limine.conf && \
|
sudo grep -qP "^\\s*///$1\\s*│" /boot/limine.conf 2>/dev/null && \
|
||||||
sudo grep -P "^\\s*///$1\\s*│" /boot/limine.conf || df_print_warning "Not found"
|
sudo grep -P "^\\s*///$1\\s*│" /boot/limine.conf || df_print_warning "Not found"
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,13 +75,13 @@ snap-delete() {
|
|||||||
[[ -z "$1" ]] && { echo "Usage: snap-delete <num>"; return 1; }
|
[[ -z "$1" ]] && { echo "Usage: snap-delete <num>"; return 1; }
|
||||||
df_print_func_name "Delete Snapshot #$1"
|
df_print_func_name "Delete Snapshot #$1"
|
||||||
|
|
||||||
local before=$(sudo grep -cP "^\\s*///\\d+\\s*│" /boot/limine.conf || echo "0")
|
local before=$(sudo grep -cP "^\\s*///\\d+\\s*│" /boot/limine.conf 2>/dev/null || echo "0")
|
||||||
sudo snapper -c root delete "$1" && df_print_success "Deleted #$1" || { df_print_error "Failed"; return 1; }
|
sudo snapper -c root delete "$1" && df_print_success "Deleted #$1" || { df_print_error "Failed"; return 1; }
|
||||||
|
|
||||||
df_print_step "Syncing limine..."
|
df_print_step "Syncing limine..."
|
||||||
sudo systemctl start limine-snapper-sync.service; sleep 2
|
sudo systemctl start limine-snapper-sync.service; sleep 2
|
||||||
|
|
||||||
sudo grep -qP "^\\s*///$1\\s*│" /boot/limine.conf && df_print_error "Still in limine!" || df_print_success "Removed from limine"
|
sudo grep -qP "^\\s*///$1\\s*│" /boot/limine.conf 2>/dev/null && df_print_error "Still in limine!" || df_print_success "Removed from limine"
|
||||||
}
|
}
|
||||||
|
|
||||||
snap-sync() {
|
snap-sync() {
|
||||||
@@ -89,10 +95,235 @@ snap-check() {
|
|||||||
local latest=$(sudo snapper -c root list | tail -n +3 | grep -v "^\s*0\s" | tail -1 | awk '{print $1}')
|
local latest=$(sudo snapper -c root list | tail -n +3 | grep -v "^\s*0\s" | tail -1 | awk '{print $1}')
|
||||||
[[ -z "$latest" ]] && { df_print_warning "No snapshots"; return 1; }
|
[[ -z "$latest" ]] && { df_print_warning "No snapshots"; return 1; }
|
||||||
df_print_info "Latest: #$latest"
|
df_print_info "Latest: #$latest"
|
||||||
sudo grep -qP "^\\s*///$latest\\s*│" /boot/limine.conf && \
|
sudo grep -qP "^\\s*///$latest\\s*│" /boot/limine.conf 2>/dev/null && \
|
||||||
df_print_success "Latest in limine.conf" || df_print_error "Latest NOT in limine.conf"
|
df_print_success "Latest in limine.conf" || df_print_error "Latest NOT in limine.conf"
|
||||||
local count=$(sudo grep -cP "^\\s*///\\d+\\s*│" /boot/limine.conf || echo "0")
|
local count=$(sudo grep -cP "^\\s*///\\d+\\s*│" /boot/limine.conf 2>/dev/null || echo "0")
|
||||||
df_print_info "Total entries: $count"
|
df_print_info "Total entries: $count"
|
||||||
}
|
}
|
||||||
|
|
||||||
alias snap='snap-create' snapls='snap-list' snaprm='snap-delete' snapcheck='snap-check'
|
# ============================================================================
|
||||||
|
# System Update with PRE/POST Snapshots
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
sys-update() {
|
||||||
|
local update_date=$(date +"%Y-%m-%d %H:%M")
|
||||||
|
local limine="/boot/limine.conf"
|
||||||
|
local pre_num=""
|
||||||
|
local post_num=""
|
||||||
|
local update_cmd=""
|
||||||
|
local update_success=false
|
||||||
|
|
||||||
|
df_print_func_name "System Update with Snapshots"
|
||||||
|
|
||||||
|
# Check for snapper
|
||||||
|
if ! command -v snapper &>/dev/null; then
|
||||||
|
df_print_warning "Snapper not installed, running update without snapshots"
|
||||||
|
_sys_update_run
|
||||||
|
return $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Determine update command
|
||||||
|
if command -v paru &>/dev/null; then
|
||||||
|
update_cmd="paru -Syu"
|
||||||
|
elif command -v yay &>/dev/null; then
|
||||||
|
update_cmd="yay -Syu"
|
||||||
|
else
|
||||||
|
update_cmd="sudo pacman -Syu"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
df_print_info "Update command: $update_cmd"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# PRE Snapshot
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
df_print_step "Creating PRE snapshot..."
|
||||||
|
|
||||||
|
pre_num=$(sudo snapper -c root create \
|
||||||
|
--type pre \
|
||||||
|
--cleanup-algorithm number \
|
||||||
|
--description "System Update PRE ${update_date}" \
|
||||||
|
--print-number 2>/dev/null)
|
||||||
|
|
||||||
|
if [[ -z "$pre_num" || ! "$pre_num" =~ ^[0-9]+$ ]]; then
|
||||||
|
df_print_error "Failed to create PRE snapshot"
|
||||||
|
echo ""
|
||||||
|
read -q "REPLY?Continue update without snapshots? [y/N] "
|
||||||
|
echo ""
|
||||||
|
[[ "$REPLY" != "y" ]] && { df_print_info "Aborted"; return 1; }
|
||||||
|
_sys_update_run
|
||||||
|
return $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
df_print_success "PRE snapshot: #$pre_num"
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Run Update
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
echo ""
|
||||||
|
df_print_step "Running system update..."
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Run the update (don't use sudo if using paru/yay)
|
||||||
|
if [[ "$update_cmd" == "sudo pacman -Syu" ]]; then
|
||||||
|
sudo pacman -Syu
|
||||||
|
else
|
||||||
|
eval "$update_cmd"
|
||||||
|
fi
|
||||||
|
|
||||||
|
local exit_code=$?
|
||||||
|
|
||||||
|
if [[ $exit_code -eq 0 ]]; then
|
||||||
|
update_success=true
|
||||||
|
df_print_success "Update completed successfully"
|
||||||
|
else
|
||||||
|
df_print_warning "Update finished with exit code: $exit_code"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# POST Snapshot
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
echo ""
|
||||||
|
df_print_step "Creating POST snapshot..."
|
||||||
|
|
||||||
|
post_num=$(sudo snapper -c root create \
|
||||||
|
--type post \
|
||||||
|
--cleanup-algorithm number \
|
||||||
|
--pre-number "$pre_num" \
|
||||||
|
--description "System Update POST ${update_date}" \
|
||||||
|
--print-number 2>/dev/null)
|
||||||
|
|
||||||
|
if [[ -z "$post_num" || ! "$post_num" =~ ^[0-9]+$ ]]; then
|
||||||
|
df_print_error "Failed to create POST snapshot"
|
||||||
|
df_print_warning "PRE snapshot #$pre_num exists without POST pair"
|
||||||
|
else
|
||||||
|
df_print_success "POST snapshot: #$post_num (linked to PRE #$pre_num)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Sync with Limine
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
echo ""
|
||||||
|
df_print_step "Syncing with limine bootloader..."
|
||||||
|
|
||||||
|
if [[ -f "$limine" ]]; then
|
||||||
|
sudo systemctl start limine-snapper-sync.service 2>/dev/null
|
||||||
|
sleep 2
|
||||||
|
|
||||||
|
if [[ -n "$post_num" ]] && sudo grep -qP "^\\s*///$post_num\\s*│" "$limine" 2>/dev/null; then
|
||||||
|
df_print_success "Snapshot #$post_num added to boot menu"
|
||||||
|
else
|
||||||
|
df_print_info "Limine sync triggered (may take a moment)"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
df_print_info "Limine not detected, skipping boot menu sync"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Update package count for prompt
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
if command -v checkupdates &>/dev/null; then
|
||||||
|
export UPDATE_PKG_COUNT=$(checkupdates 2>/dev/null | wc -l)
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Summary
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
echo ""
|
||||||
|
df_print_section "Summary"
|
||||||
|
df_print_indent "PRE snapshot: #$pre_num"
|
||||||
|
[[ -n "$post_num" ]] && df_print_indent "POST snapshot: #$post_num"
|
||||||
|
df_print_indent "Status: $([[ "$update_success" == true ]] && echo "Success" || echo "Completed with warnings")"
|
||||||
|
echo ""
|
||||||
|
df_print_info "To rollback: sudo snapper -c root undochange $pre_num..$post_num"
|
||||||
|
df_print_info "To compare: sudo snapper -c root status $pre_num..$post_num"
|
||||||
|
|
||||||
|
return $exit_code
|
||||||
|
}
|
||||||
|
|
||||||
|
# Helper for running update without snapshots
|
||||||
|
_sys_update_run() {
|
||||||
|
if command -v paru &>/dev/null; then
|
||||||
|
paru -Syu
|
||||||
|
elif command -v yay &>/dev/null; then
|
||||||
|
yay -Syu
|
||||||
|
else
|
||||||
|
sudo pacman -Syu
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Update package count for prompt
|
||||||
|
if command -v checkupdates &>/dev/null; then
|
||||||
|
export UPDATE_PKG_COUNT=$(checkupdates 2>/dev/null | wc -l)
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# Rollback Helper
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
sys-rollback() {
|
||||||
|
df_print_func_name "System Rollback"
|
||||||
|
|
||||||
|
# Show recent pre/post pairs
|
||||||
|
df_print_step "Recent update snapshots:"
|
||||||
|
echo ""
|
||||||
|
sudo snapper -c root list --type pre-post 2>/dev/null | tail -10
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
if [[ -z "$1" ]]; then
|
||||||
|
df_print_info "Usage: sys-rollback <pre-number>"
|
||||||
|
df_print_info "Example: sys-rollback 42"
|
||||||
|
echo ""
|
||||||
|
df_print_info "This will undo changes between the PRE and POST snapshots."
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
local pre_num="$1"
|
||||||
|
|
||||||
|
# Find the corresponding POST snapshot
|
||||||
|
local post_num=$(sudo snapper -c root list --type pre-post 2>/dev/null | \
|
||||||
|
awk -v pre="$pre_num" '$1 == pre {print $2}')
|
||||||
|
|
||||||
|
if [[ -z "$post_num" ]]; then
|
||||||
|
df_print_error "Could not find POST snapshot for PRE #$pre_num"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
df_print_warning "This will undo all changes between snapshot #$pre_num and #$post_num"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Show what would change
|
||||||
|
df_print_step "Changes to be reverted:"
|
||||||
|
sudo snapper -c root status "$pre_num..$post_num" 2>/dev/null | head -20
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
read -q "REPLY?Proceed with rollback? [y/N] "
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
if [[ "$REPLY" == "y" ]]; then
|
||||||
|
df_print_step "Rolling back..."
|
||||||
|
sudo snapper -c root undochange "$pre_num..$post_num"
|
||||||
|
|
||||||
|
if [[ $? -eq 0 ]]; then
|
||||||
|
df_print_success "Rollback complete"
|
||||||
|
df_print_warning "Reboot recommended to apply changes"
|
||||||
|
else
|
||||||
|
df_print_error "Rollback failed"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
df_print_info "Rollback cancelled"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# ============================================================================
|
||||||
|
# Aliases
|
||||||
|
# ============================================================================
|
||||||
|
|
||||||
|
alias snap='snap-create'
|
||||||
|
alias snapls='snap-list'
|
||||||
|
alias snaprm='snap-delete'
|
||||||
|
alias snapcheck='snap-check'
|
||||||
|
alias snapsync='snap-sync'
|
||||||
|
alias rollback='sys-rollback'
|
||||||
|
|||||||
Reference in New Issue
Block a user