mirror of
https://github.com/tw93/Mole.git
synced 2026-02-04 17:24:45 +00:00
226 lines
7.0 KiB
Bash
226 lines
7.0 KiB
Bash
#!/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 exists
|
|
mkdir -p "$(dirname "$LOG_FILE")" 2> /dev/null || true
|
|
|
|
# ============================================================================
|
|
# Log Rotation
|
|
# ============================================================================
|
|
|
|
# Rotate log file if it exceeds max size
|
|
# Called once at module load, not per log entry
|
|
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
|
|
touch "$LOG_FILE" 2> /dev/null || true
|
|
fi
|
|
}
|
|
|
|
# ============================================================================
|
|
# Logging Functions
|
|
# ============================================================================
|
|
|
|
# Log informational message
|
|
# Args: $1 - 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
|
|
# Args: $1 - 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
|
|
# Args: $1 - 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
|
|
# Args: $1 - message
|
|
log_error() {
|
|
echo -e "${RED}${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 - only shown when MO_DEBUG=1
|
|
# Args: $@ - debug message components
|
|
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
|
|
: > "$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
|
|
# Args: $@ - command and arguments
|
|
run_silent() {
|
|
"$@" > /dev/null 2>&1 || true
|
|
}
|
|
|
|
# Run command with error logging
|
|
# Args: $@ - command and arguments
|
|
# Returns: command exit code
|
|
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 with heading and details
|
|
# Args: $1=status (ignored), $2=heading, $@=details
|
|
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
|