#!/bin/bash
# --- Загрузка конфига ---
CONFIG_FILE="/srv/smart_monitor.conf"
if [ -f "$CONFIG_FILE" ]; then
. "$CONFIG_FILE"
else
echo "Config file $CONFIG_FILE not found!" >&2
exit 1
fi
# --- Load library ---
LIBRARY_FILE="/srv/lib_common.sh"
if [ -f "$LIBRARY_FILE" ]; then
. "$LIBRARY_FILE"
else
echo "Library file $LIBRARY_FILE not found!" >&2
exit 1
fi
# --- Настройки ---
LOG="${LOG_FILE:-/var/log/smart_check.log}"
START_TIME=$(date "+%Y-%m-%d %H:%M:%S")
{
echo "=========================================="
echo "Smart check started at $START_TIME"
echo ""
echo "Config loaded from: $CONFIG_FILE"
echo ""
echo "--- Disk Check Results ---"
} >> "$LOG"
# --- Поиск дисков (исключаем виртуальные устройства) ---
DISKS_HDD=$(lsblk -d -o NAME,ROTA,TYPE | awk '$2 == "1" && $3 == "disk" && $1 !~ /^loop|^sr/ {print "/dev/"$1}')
DISKS_NVME=$(lsblk -d -o NAME,TYPE | awk '$1 ~ /^nvme/ && $2 == "disk" {print "/dev/"$1}')
DISKS_SSD=$(comm -23 \
<(lsblk -d -o NAME,ROTA,TYPE | awk '$2 == "0" && $3 == "disk" && $1 !~ /^loop|^nvme|^sr/ {print "/dev/"$1}' | sort) \
<(echo "$DISKS_NVME" | sort))
# --- Проверка диска ---
check_disk() {
local disk=$1
local disk_type=$2
local -A result=(
["disk"]="$disk"
["type"]="$disk_type"
["status"]="UNKNOWN"
["errors"]=""
)
local smart_data=$(sudo smartctl -H -A "$disk" 2>/dev/null)
if [[ "$disk_type" == "NVMe" ]]; then
result["status"]=$(echo "$smart_data" | grep -i "SMART overall-health" | awk '{print $NF}')
# NVMe-специфичные атрибуты
result["media_errors"]=$(echo "$smart_data" | grep -i "Media and Data Integrity Errors" | awk '{print $NF}')
result["log_errors"]=$(echo "$smart_data" | grep -i "Error Information Log Entries" | awk '{print $NF}')
[[ "${result["media_errors"]}" =~ ^[0-9]+$ ]] || result["media_errors"]=0
[[ "${result["log_errors"]}" =~ ^[0-9]+$ ]] || result["log_errors"]=0
[ "${result["media_errors"]}" -ne 0 ] && result["errors"]+="Media Errors: ${result["media_errors"]} "
[ "${result["log_errors"]}" -ne 0 ] && result["errors"]+="Log Errors: ${result["log_errors"]} "
else
result["status"]=$(echo "$smart_data" | grep -i "test result" | awk '{print $NF}')
# Атрибуты для HDD/SATA SSD
while read -r line; do
key=$(echo "$line" | awk '{print $1}')
value=$(echo "$line" | awk '{print $10}')
[ "$value" != "0" ] && result["errors"]+="$key=$value "
done <<< "$(echo "$smart_data" | grep -E "Reallocated_Sector_Ct|Current_Pending_Sector|Uncorrectable_Error_Ct")"
fi
result["errors"]="${result["errors"]% }"
echo "$(declare -p result)"
}
# --- Основная проверка ---
declare -a ALL_RESULTS
ERRORS_FOUND=0
#{
# echo "--- Проверка дисков ---"
# echo "Типы дисков:"
# echo "HDD: $DISKS_HDD"
# echo "SATA SSD: $DISKS_SSD"
# echo "NVMe: $DISKS_NVME"
# echo ""
#} >> "$LOG"
for disk in $DISKS_HDD $DISKS_SSD $DISKS_NVME; do
if [[ "$DISKS_HDD" == *"$disk"* ]]; then
disk_type="HDD"
elif [[ "$DISKS_SSD" == *"$disk"* ]]; then
disk_type="SATA SSD"
else
disk_type="NVMe"
fi
eval "$(check_disk "$disk" "$disk_type")"
# Человеко-читаемый лог
{
printf "Диск: %-16s | Тип: %-8s | Статус: %-6s" "${result[disk]}" "${result[type]}" "${result[status]}"
[ -n "${result[errors]}" ] && printf " | Ошибки: %s" "${result[errors]}"
printf "\n"
} >> "$LOG"
ALL_RESULTS+=("$(declare -p result)")
[[ "${result[status]}" != "PASSED" ]] && ERRORS_FOUND=1
done
# --- Уведомление в Telegram ---
TELEGRAM_STATUS="Not sent"
if [[ "$TELEGRAM_POST_MESSAGE_ALWAYS" == "1" || "$ERRORS_FOUND" == "1" ]]; then
# Формируем сообщение с HTML-разметкой
MESSAGE="🛡️ SMART Report | $(date '+%Y-%m-%d %H:%M:%S')%0A%0A"
MESSAGE+="Host: $(hostname)%0A%0A"
for result_str in "${ALL_RESULTS[@]}"; do
eval "$result_str"
if [[ "${result[status]}" == "PASSED" ]]; then
MESSAGE+="✅ ${result[type]}: ${result[disk]}
(PASSED)%0A"
else
MESSAGE+="🔴 ${result[type]}: ${result[disk]}
(${result[status]})%0A"
if [[ -n "${result[errors]}" ]]; then
MESSAGE+="Errors: ${result[errors]// /
}
%0A"
fi
MESSAGE+="%0A"
fi
done
# Отправляем в Telegram
TELEGRAM_STATUS=$(send_to_telegram "$TELEGRAM_BOT_TOKEN" "$TELEGRAM_CHAT_ID" "$MESSAGE")
fi
# --- Финальный лог ---
{
echo ""
echo "--- Notification Status ---"
echo "Telegram: $TELEGRAM_STATUS"
echo ""
echo "Finished at: $(date '+%Y-%m-%d %H:%M:%S')"
echo "=========================================="
echo ""
} >> "$LOG"