#!/bin/bash # Mole - Logging System # Centralized logging with rotation support set -euo pipefail # Prevent multiple sourcing if [[ -n "${MOLE_LOG_LOADED:-}" ]]; then return 0 fi readonly MOLE_LOG_LOADED=1 # Ensure base.sh is loaded for colors and icons if [[ -z "${MOLE_BASE_LOADED:-}" ]]; then _MOLE_CORE_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" # shellcheck source=lib/core/base.sh source "$_MOLE_CORE_DIR/base.sh" fi # ============================================================================ # Logging Configuration # ============================================================================ readonly LOG_FILE="${HOME}/.config/mole/mole.log" readonly DEBUG_LOG_FILE="${HOME}/.config/mole/mole_debug_session.log" readonly LOG_MAX_SIZE_DEFAULT=1048576 # 1MB # Ensure log directory and file exist with correct ownership ensure_user_file "$LOG_FILE" # ============================================================================ # Log Rotation # ============================================================================ # Rotate log file if it exceeds maximum size rotate_log_once() { # Skip if already checked this session [[ -n "${MOLE_LOG_ROTATED:-}" ]] && return 0 export MOLE_LOG_ROTATED=1 local max_size="${MOLE_MAX_LOG_SIZE:-$LOG_MAX_SIZE_DEFAULT}" if [[ -f "$LOG_FILE" ]] && [[ $(get_file_size "$LOG_FILE") -gt "$max_size" ]]; then mv "$LOG_FILE" "${LOG_FILE}.old" 2> /dev/null || true ensure_user_file "$LOG_FILE" fi } # ============================================================================ # Logging Functions # ============================================================================ # Log informational message log_info() { echo -e "${BLUE}$1${NC}" local timestamp=$(date '+%Y-%m-%d %H:%M:%S') echo "[$timestamp] INFO: $1" >> "$LOG_FILE" 2> /dev/null || true if [[ "${MO_DEBUG:-}" == "1" ]]; then echo "[$timestamp] INFO: $1" >> "$DEBUG_LOG_FILE" 2> /dev/null || true fi } # Log success message log_success() { echo -e " ${GREEN}${ICON_SUCCESS}${NC} $1" local timestamp=$(date '+%Y-%m-%d %H:%M:%S') echo "[$timestamp] SUCCESS: $1" >> "$LOG_FILE" 2> /dev/null || true if [[ "${MO_DEBUG:-}" == "1" ]]; then echo "[$timestamp] SUCCESS: $1" >> "$DEBUG_LOG_FILE" 2> /dev/null || true fi } # Log warning message log_warning() { echo -e "${YELLOW}$1${NC}" local timestamp=$(date '+%Y-%m-%d %H:%M:%S') echo "[$timestamp] WARNING: $1" >> "$LOG_FILE" 2> /dev/null || true if [[ "${MO_DEBUG:-}" == "1" ]]; then echo "[$timestamp] WARNING: $1" >> "$DEBUG_LOG_FILE" 2> /dev/null || true fi } # Log error message log_error() { echo -e "${YELLOW}${ICON_ERROR}${NC} $1" >&2 local timestamp=$(date '+%Y-%m-%d %H:%M:%S') echo "[$timestamp] ERROR: $1" >> "$LOG_FILE" 2> /dev/null || true if [[ "${MO_DEBUG:-}" == "1" ]]; then echo "[$timestamp] ERROR: $1" >> "$DEBUG_LOG_FILE" 2> /dev/null || true fi } # Debug logging (active when MO_DEBUG=1) debug_log() { if [[ "${MO_DEBUG:-}" == "1" ]]; then echo -e "${GRAY}[DEBUG]${NC} $*" >&2 echo "[$(date '+%Y-%m-%d %H:%M:%S')] DEBUG: $*" >> "$DEBUG_LOG_FILE" 2> /dev/null || true fi } # Log system information for debugging log_system_info() { # Only allow once per session [[ -n "${MOLE_SYS_INFO_LOGGED:-}" ]] && return 0 export MOLE_SYS_INFO_LOGGED=1 # Reset debug log file for this new session ensure_user_file "$DEBUG_LOG_FILE" : > "$DEBUG_LOG_FILE" # Start block in debug log file { echo "----------------------------------------------------------------------" echo "Mole Debug Session - $(date '+%Y-%m-%d %H:%M:%S')" echo "----------------------------------------------------------------------" echo "User: $USER" echo "Hostname: $(hostname)" echo "Architecture: $(uname -m)" echo "Kernel: $(uname -r)" if command -v sw_vers > /dev/null; then echo "macOS: $(sw_vers -productVersion) ($(sw_vers -buildVersion))" fi echo "Shell: ${SHELL:-unknown} (${TERM:-unknown})" # Check sudo status non-interactively if sudo -n true 2> /dev/null; then echo "Sudo Access: Active" else echo "Sudo Access: Required" fi echo "----------------------------------------------------------------------" } >> "$DEBUG_LOG_FILE" 2> /dev/null || true # Notification to stderr echo -e "${GRAY}[DEBUG] Debug logging enabled. Session log: $DEBUG_LOG_FILE${NC}" >&2 } # ============================================================================ # Command Execution Wrappers # ============================================================================ # Run command silently (ignore errors) run_silent() { "$@" > /dev/null 2>&1 || true } # Run command with error logging run_logged() { local cmd="$1" # Log to main file, and also to debug file if enabled if [[ "${MO_DEBUG:-}" == "1" ]]; then if ! "$@" 2>&1 | tee -a "$LOG_FILE" | tee -a "$DEBUG_LOG_FILE" > /dev/null; then log_warning "Command failed: $cmd" return 1 fi else if ! "$@" 2>&1 | tee -a "$LOG_FILE" > /dev/null; then log_warning "Command failed: $cmd" return 1 fi fi return 0 } # ============================================================================ # Formatted Output # ============================================================================ # Print formatted summary block print_summary_block() { local heading="" local -a details=() local saw_heading=false # Parse arguments for arg in "$@"; do if [[ "$saw_heading" == "false" ]]; then saw_heading=true heading="$arg" else details+=("$arg") fi done local divider="======================================================================" # Print with dividers echo "" echo "$divider" if [[ -n "$heading" ]]; then echo -e "${BLUE}${heading}${NC}" fi # Print details for detail in "${details[@]}"; do [[ -z "$detail" ]] && continue echo -e "${detail}" done echo "$divider" # If debug mode is on, remind user about the log file location if [[ "${MO_DEBUG:-}" == "1" ]]; then echo -e "${GRAY}Debug session log saved to:${NC} ${DEBUG_LOG_FILE}" fi } # ============================================================================ # Initialize Logging # ============================================================================ # Perform log rotation check on module load rotate_log_once # If debug mode is enabled, log system info immediately if [[ "${MO_DEBUG:-}" == "1" ]]; then log_system_info fi