-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathsmart_report.sh
executable file
·131 lines (117 loc) · 6.83 KB
/
smart_report.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
#!/usr/bin/env bash
#
# Send a SMART status summary and detailed report of all SATA drives via Email.
readonly SCRIPT_PATH="$(cd "$(dirname "${BASH_SOURCE[0]}")" &>/dev/null && pwd)"
# shellcheck source=user.example.conf
source "${SCRIPT_PATH}/user.conf"
# shellcheck source=global.conf
source "${SCRIPT_PATH}/global.conf"
# shellcheck source=format_email.sh
source "${SCRIPT_PATH}/format_email.sh"
readonly EMAIL_SUBJECT="TrueNAS $(hostname): SMART status report"
readonly EMAIL_BODY="/tmp/smart_report.html"
# Only specify monospace font to let Email client decide of the rest.
echo "<pre style=\"font-family:monospace\">" > "${EMAIL_BODY}"
# Print a summary table of the status of all drives.
(
echo "<b>SMART status report summary for all drives:</b>"
echo "+------+---------------+----+-----+-----+-----+-------+-------+--------+------+------+------+-------+----+"
echo "|Device|Serial |Temp|Power|Start|Spin |ReAlloc|Current|Offline |UDMA |Seek |High |Command|Last|"
echo "| | | |On |Stop |Retry|Sectors|Pending|Uncorrec|CRC |Errors|Fly |Timeout|Test|"
echo "| | | |Hours|Count|Count| |Sectors|Sectors |Errors| |Writes|Count |Age |"
echo "+------+---------------+----+-----+-----+-----+-------+-------+--------+------+------+------+-------+----+"
) >> "${EMAIL_BODY}"
for drive_label in ${SATA_DRIVES}; do
# Ask smartctl to diplay the Seek_Error_Rate in raw hexadecimal so that we can extract the number of seek errors and
# total number of seeks afterwards.
drive_status="$(smartctl -A -i -v 7,hex48 /dev/"${drive_label}")"
drive_tests_list="$(smartctl -l selftest /dev/"${drive_label}")"
last_test_hours="$(echo "${drive_tests_list}" | grep "# 1" | awk '{print $9}')"
serial_number="$(echo "${drive_status}" | grep "Serial Number:" | awk '{print $3}')"
temperature="$(echo "${drive_status}" | grep "Temperature_Celsius" | awk '{print $10}')" # SATA HDD
if [[ -z "${temperature}" ]]; then
temperature="$(echo "${drive_status}" | grep "Airflow_Temperature_Cel" | awk '{print $10}')" # SATA SSD
fi
power_on_hours="$(echo "${drive_status}" | grep "Power_On_Hours" | awk '{print $10}')"
start_stop_count="$(echo "${drive_status}" | grep "Start_Stop_Count" | awk '{print $10}')"
spin_retry_count="$(echo "${drive_status}" | grep "Spin_Retry_Count" | awk '{print $10}')"
realocated_sectors="$(echo "${drive_status}" | grep "Reallocated_Sector" | awk '{print $10}')"
pending_sectors_count="$(echo "${drive_status}" | grep "Current_Pending_Sector" | awk '{print $10}')"
uncorrectable_sectors_count="$(echo "${drive_status}" | grep "Offline_Uncorrectable" | awk '{print $10}')"
udma_crc_errors_count="$(echo "${drive_status}" | grep "UDMA_CRC_Error_Count" | awk '{print $10}')"
# Using cut to grab the first 4 hex symbols which indicate the actual number of seek errors.
seek_errors="$(echo "${drive_status}" | grep "Seek_Error_Rate" | awk '{print $10}' | cut -c 1-6)"
# Using cut to grab the last 8 hex symbols which indicate the total number of seeks.
total_seeks="$(echo "${drive_status}" | grep "Seek_Error_Rate" | awk '{print $10}' | cut -c 1-2,7-14)"
high_fly_writes="$(echo "${drive_status}" | grep "High_Fly_Writes" | awk '{print $10}')"
command_timeout="$(echo "${drive_status}" | grep "Command_Timeout" | awk '{print $10}')"
# Force LC_NUMERIC because on certain non en_US systems the decimal separator is a comma and we need a dot.
# printf "%.0f" in order to round the resulting number.
# Bash doesn't "natively" support float numbers so bc is used to have a float result to a division.
test_age="$(LC_NUMERIC="en_US.UTF-8" printf "%.0f\n" "$(bc <<<"scale=6; (${power_on_hours} - ${last_test_hours}) / 24")")"
# Choose the symbol to display beside the drive name.
if [[ "${temperature}" -ge "${DRIVE_TEMPERATURE_CRITICAL}" ]] ||
[[ "${realocated_sectors}" -gt "${DRIVE_SECTORS_CRITICAL}" ]] ||
[[ "${pending_sectors_count}" -gt "${DRIVE_SECTORS_CRITICAL}" ]] ||
[[ "${uncorrectable_sectors_count}" -gt "${DRIVE_SECTORS_CRITICAL}" ]]; then
ui_symbol="${UI_CRITICAL_SYMBOL}"
elif [[ "${temperature}" -ge "${DRIVE_TEMPERATURE_WARNING}" ]] ||
[[ "${realocated_sectors}" -gt "0" ]] ||
[[ "${pending_sectors_count}" -gt "0" ]] ||
[[ "${uncorrectable_sectors_count}" -gt "0" ]] ||
[[ "${test_age}" -ge "${SMART_TEST_AGE_WARNING}" ]]; then
ui_symbol="${UI_WARNING_SYMBOL}"
else
ui_symbol=" "
fi
# seek_errors and total_seeks are stored as hex values, we need to convert them before comparing and displaying.
seek_errors="$(printf "%d" "${seek_errors}")"
total_seeks="$(printf "%d" "${total_seeks}")"
# If there's no seeks at all it means the parameter is not supported by the drive SMART so display "N/A".
if [[ "${total_seeks}" == "0" ]]; then
seek_errors="N/A"
total_seeks="N/A"
fi
# Same for those two parameters.
if [[ "${high_fly_writes}" == "" ]]; then
high_fly_writes="N/A"
fi
if [[ "${command_timeout}" == "" ]]; then
command_timeout="N/A"
fi
# Print the row with all the attributes corresponding to the drive.
printf "|%-4s %1s|%-15s| %s |%5s|%5s|%5s|%7s|%7s|%8s|%6s|%6s|%6s|%7s|%4s|\n" "${drive_label}" "${ui_symbol}" \
"${serial_number}" "${temperature}" "${power_on_hours}" "${start_stop_count}" "${spin_retry_count}" \
"${realocated_sectors}" "${pending_sectors_count}" "${uncorrectable_sectors_count}" "${udma_crc_errors_count}" \
"${seek_errors}" "${high_fly_writes}" "${command_timeout}" "${test_age}" >> "${EMAIL_BODY}"
done
echo "+------+---------------+----+-----+-----+-----+-------+-------+--------+------+------+------+-------+----+" >> "${EMAIL_BODY}"
# Print a detailed SMART report for each drive.
for drive_label in ${SATA_DRIVES}; do
drive_smart_info="$(smartctl -i /dev/"${drive_label}")"
brand="$(echo "${drive_smart_info}" | grep "Model Family" | awk '{print $3, $4, $5}')"
serial_number="$(echo "${drive_smart_info}" | grep "Serial Number" | awk '{print $3}')"
(
echo ""
echo ""
echo "<b>SMART status report for ${drive_label} drive (${brand}: ${serial_number}):</b>"
# Dislpay the SMART status table.
smartctl -H -A -l error /dev/"${drive_label}"
# Display the status of the last selftest.
smartctl -l selftest /dev/"${drive_label}" | grep "# 1 \|Num" | cut -c6-
) >> "${EMAIL_BODY}"
done
# Trimming unnecessary information from SMART detailed reports.
sed -i '' -e '/smartctl 6.3/d' "${EMAIL_BODY}"
sed -i '' -e '/Copyright/d' "${EMAIL_BODY}"
sed -i '' -e '/=== START OF READ/d' "${EMAIL_BODY}"
sed -i '' -e '/SMART Attributes Data/d' "${EMAIL_BODY}"
sed -i '' -e '/Vendor Specific SMART/d' "${EMAIL_BODY}"
sed -i '' -e '/SMART Error Log Version/d' "${EMAIL_BODY}"
(
echo ""
echo "-- End of SMART status report --"
echo "</pre>"
) >> "${EMAIL_BODY}"
format_email "${EMAIL_SUBJECT}" "${EMAIL_ADDRESS}" "${EMAIL_BODY}" | sendmail -i -t
rm "${EMAIL_BODY}"