bench.sh 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. #!/usr/bin/env bash
  2. #
  3. # Description: A Bench Script by Teddysun
  4. #
  5. # Copyright (C) 2015 - 2025 Teddysun <[email protected]>
  6. # Thanks: LookBack <[email protected]>
  7. # URL: https://teddysun.com/444.html
  8. # https://github.com/teddysun/across/blob/master/bench.sh
  9. #
  10. trap _exit INT QUIT TERM
  11. _red() {
  12. printf '\033[0;31;31m%b\033[0m' "$1"
  13. }
  14. _green() {
  15. printf '\033[0;31;32m%b\033[0m' "$1"
  16. }
  17. _yellow() {
  18. printf '\033[0;31;33m%b\033[0m' "$1"
  19. }
  20. _blue() {
  21. printf '\033[0;31;36m%b\033[0m' "$1"
  22. }
  23. _exists() {
  24. local cmd="$1"
  25. if eval type type >/dev/null 2>&1; then
  26. eval type "$cmd" >/dev/null 2>&1
  27. elif command >/dev/null 2>&1; then
  28. command -v "$cmd" >/dev/null 2>&1
  29. else
  30. which "$cmd" >/dev/null 2>&1
  31. fi
  32. local rt=$?
  33. return ${rt}
  34. }
  35. _exit() {
  36. _red "\nThe script has been terminated. Cleaning up files...\n"
  37. # clean up
  38. rm -fr speedtest.tgz speedtest-cli benchtest_*
  39. exit 1
  40. }
  41. get_opsy() {
  42. [ -f /etc/redhat-release ] && awk '{print $0}' /etc/redhat-release && return
  43. [ -f /etc/os-release ] && awk -F'[= "]' '/PRETTY_NAME/{print $3,$4,$5}' /etc/os-release && return
  44. [ -f /etc/lsb-release ] && awk -F'[="]+' '/DESCRIPTION/{print $2}' /etc/lsb-release && return
  45. }
  46. next() {
  47. printf "%-70s\n" "-" | sed 's/\s/-/g'
  48. }
  49. speed_test() {
  50. local nodeName="$2"
  51. if [ -z "$1" ];then
  52. ./speedtest-cli/speedtest --progress=no --accept-license --accept-gdpr >./speedtest-cli/speedtest.log 2>&1
  53. else
  54. ./speedtest-cli/speedtest --progress=no --server-id="$1" --accept-license --accept-gdpr >./speedtest-cli/speedtest.log 2>&1
  55. fi
  56. if [ $? -eq 0 ]; then
  57. local dl_speed up_speed latency
  58. dl_speed=$(awk '/Download/{print $3" "$4}' ./speedtest-cli/speedtest.log)
  59. up_speed=$(awk '/Upload/{print $3" "$4}' ./speedtest-cli/speedtest.log)
  60. latency=$(awk '/Latency/{print $3" "$4}' ./speedtest-cli/speedtest.log)
  61. if [[ -n "${dl_speed}" && -n "${up_speed}" && -n "${latency}" ]]; then
  62. printf "\033[0;33m%-18s\033[0;32m%-18s\033[0;31m%-20s\033[0;36m%-12s\033[0m\n" " ${nodeName}" "${up_speed}" "${dl_speed}" "${latency}"
  63. fi
  64. fi
  65. }
  66. speed() {
  67. speed_test '' 'Speedtest.net'
  68. speed_test '21541' 'Los Angeles, US'
  69. speed_test '43860' 'Dallas, US'
  70. speed_test '40879' 'Montreal, CA'
  71. speed_test '61933' 'Paris, FR'
  72. speed_test '28922' 'Amsterdam, NL'
  73. speed_test '25858' 'Beijing, CN'
  74. speed_test '24447' 'Shanghai, CN'
  75. speed_test '60572' 'Guangzhou, CN'
  76. speed_test '32155' 'Hong Kong, CN'
  77. speed_test '13623' 'Singapore, SG'
  78. speed_test '48463' 'Tokyo, JP'
  79. }
  80. io_test() {
  81. (LANG=C dd if=/dev/zero of=benchtest_$$ bs=512k count="$1" conv=fdatasync && rm -f benchtest_$$) 2>&1 | awk -F '[,,]' '{io=$NF} END { print io}' | sed 's/^[ \t]*//;s/[ \t]*$//'
  82. }
  83. calc_size() {
  84. local raw=$1
  85. local total_size=0
  86. local num=1
  87. local unit="KB"
  88. if ! [[ ${raw} =~ ^[0-9]+$ ]]; then
  89. echo ""
  90. return
  91. fi
  92. if [ "${raw}" -ge 1073741824 ]; then
  93. num=1073741824
  94. unit="TB"
  95. elif [ "${raw}" -ge 1048576 ]; then
  96. num=1048576
  97. unit="GB"
  98. elif [ "${raw}" -ge 1024 ]; then
  99. num=1024
  100. unit="MB"
  101. elif [ "${raw}" -eq 0 ]; then
  102. echo "${total_size}"
  103. return
  104. fi
  105. total_size=$(awk 'BEGIN{printf "%.1f", '"$raw"' / '$num'}')
  106. echo "${total_size} ${unit}"
  107. }
  108. # since calc_size converts kilobyte to MB, GB and TB
  109. # to_kibyte converts zfs size from bytes to kilobyte
  110. to_kibyte() {
  111. local raw=$1
  112. awk 'BEGIN{printf "%.0f", '"$raw"' / 1024}'
  113. }
  114. calc_sum() {
  115. local arr=("$@")
  116. local s
  117. s=0
  118. for i in "${arr[@]}"; do
  119. s=$((s + i))
  120. done
  121. echo ${s}
  122. }
  123. check_virt() {
  124. _exists "dmesg" && virtualx="$(dmesg 2>/dev/null)"
  125. if _exists "dmidecode"; then
  126. sys_manu="$(dmidecode -s system-manufacturer 2>/dev/null)"
  127. sys_product="$(dmidecode -s system-product-name 2>/dev/null)"
  128. sys_ver="$(dmidecode -s system-version 2>/dev/null)"
  129. else
  130. sys_manu=""
  131. sys_product=""
  132. sys_ver=""
  133. fi
  134. if grep -qa docker /proc/1/cgroup; then
  135. virt="Docker"
  136. elif grep -qa lxc /proc/1/cgroup; then
  137. virt="LXC"
  138. elif grep -qa container=lxc /proc/1/environ; then
  139. virt="LXC"
  140. elif [[ -f /proc/user_beancounters ]]; then
  141. virt="OpenVZ"
  142. elif [[ "${virtualx}" == *kvm-clock* ]]; then
  143. virt="KVM"
  144. elif [[ "${sys_product}" == *KVM* ]]; then
  145. virt="KVM"
  146. elif [[ "${sys_manu}" == *QEMU* ]]; then
  147. virt="KVM"
  148. elif [[ "${cname}" == *KVM* ]]; then
  149. virt="KVM"
  150. elif [[ "${cname}" == *QEMU* ]]; then
  151. virt="KVM"
  152. elif [[ "${virtualx}" == *"VMware Virtual Platform"* ]]; then
  153. virt="VMware"
  154. elif [[ "${sys_product}" == *"VMware Virtual Platform"* ]]; then
  155. virt="VMware"
  156. elif [[ "${virtualx}" == *"Parallels Software International"* ]]; then
  157. virt="Parallels"
  158. elif [[ "${virtualx}" == *VirtualBox* ]]; then
  159. virt="VirtualBox"
  160. elif [[ -e /proc/xen ]]; then
  161. if grep -q "control_d" "/proc/xen/capabilities" 2>/dev/null; then
  162. virt="Xen-Dom0"
  163. else
  164. virt="Xen-DomU"
  165. fi
  166. elif [ -f "/sys/hypervisor/type" ] && grep -q "xen" "/sys/hypervisor/type"; then
  167. virt="Xen"
  168. elif [[ "${sys_manu}" == *"Microsoft Corporation"* ]]; then
  169. if [[ "${sys_product}" == *"Virtual Machine"* ]]; then
  170. if [[ "${sys_ver}" == *"7.0"* || "${sys_ver}" == *"Hyper-V" ]]; then
  171. virt="Hyper-V"
  172. else
  173. virt="Microsoft Virtual Machine"
  174. fi
  175. fi
  176. else
  177. virt="Dedicated"
  178. fi
  179. }
  180. ipv4_info() {
  181. local org city country region
  182. org="$(wget -q -T10 -O- http://ipinfo.io/org)"
  183. city="$(wget -q -T10 -O- http://ipinfo.io/city)"
  184. country="$(wget -q -T10 -O- http://ipinfo.io/country)"
  185. region="$(wget -q -T10 -O- http://ipinfo.io/region)"
  186. if [[ -n "${org}" ]]; then
  187. echo " Organization : $(_blue "${org}")"
  188. fi
  189. if [[ -n "${city}" && -n "${country}" ]]; then
  190. echo " Location : $(_blue "${city} / ${country}")"
  191. fi
  192. if [[ -n "${region}" ]]; then
  193. echo " Region : $(_yellow "${region}")"
  194. fi
  195. if [[ -z "${org}" ]]; then
  196. echo " Region : $(_red "No ISP detected")"
  197. fi
  198. }
  199. install_speedtest() {
  200. if [ ! -e "./speedtest-cli/speedtest" ]; then
  201. sys_bit=""
  202. local sysarch
  203. sysarch="$(uname -m)"
  204. if [ "${sysarch}" = "unknown" ] || [ "${sysarch}" = "" ]; then
  205. sysarch="$(arch)"
  206. fi
  207. if [ "${sysarch}" = "x86_64" ]; then
  208. sys_bit="x86_64"
  209. fi
  210. if [ "${sysarch}" = "i386" ] || [ "${sysarch}" = "i686" ]; then
  211. sys_bit="i386"
  212. fi
  213. if [ "${sysarch}" = "armv8" ] || [ "${sysarch}" = "armv8l" ] || [ "${sysarch}" = "aarch64" ] || [ "${sysarch}" = "arm64" ]; then
  214. sys_bit="aarch64"
  215. fi
  216. if [ "${sysarch}" = "armv7" ] || [ "${sysarch}" = "armv7l" ]; then
  217. sys_bit="armhf"
  218. fi
  219. if [ "${sysarch}" = "armv6" ]; then
  220. sys_bit="armel"
  221. fi
  222. [ -z "${sys_bit}" ] && _red "Error: Unsupported system architecture (${sysarch}).\n" && exit 1
  223. url1="https://install.speedtest.net/app/cli/ookla-speedtest-1.2.0-linux-${sys_bit}.tgz"
  224. url2="https://dl.lamp.sh/files/ookla-speedtest-1.2.0-linux-${sys_bit}.tgz"
  225. if ! wget --no-check-certificate -q -T10 -O speedtest.tgz ${url1}; then
  226. if ! wget --no-check-certificate -q -T10 -O speedtest.tgz ${url2}; then
  227. _red "Error: Failed to download speedtest-cli.\n" && exit 1
  228. fi
  229. fi
  230. mkdir -p speedtest-cli && tar zxf speedtest.tgz -C ./speedtest-cli && chmod +x ./speedtest-cli/speedtest
  231. rm -f speedtest.tgz
  232. fi
  233. printf "%-18s%-18s%-20s%-12s\n" " Node Name" "Upload Speed" "Download Speed" "Latency"
  234. }
  235. print_intro() {
  236. echo "-------------------- A Bench.sh Script By Teddysun -------------------"
  237. echo " Version : $(_green v2025-05-08)"
  238. echo " Usage : $(_red "wget -qO- bench.sh | bash")"
  239. }
  240. # Get System information
  241. get_system_info() {
  242. cname=$(awk -F: '/model name/ {name=$2} END {print name}' /proc/cpuinfo | sed 's/^[ \t]*//;s/[ \t]*$//')
  243. cores=$(awk -F: '/^processor/ {core++} END {print core}' /proc/cpuinfo)
  244. freq=$(awk -F'[ :]' '/cpu MHz/ {print $4;exit}' /proc/cpuinfo)
  245. ccache=$(awk -F: '/cache size/ {cache=$2} END {print cache}' /proc/cpuinfo | sed 's/^[ \t]*//;s/[ \t]*$//')
  246. cpu_aes=$(grep -i 'aes' /proc/cpuinfo)
  247. cpu_virt=$(grep -Ei 'vmx|svm' /proc/cpuinfo)
  248. tram=$(
  249. LANG=C
  250. free | awk '/Mem/ {print $2}'
  251. )
  252. tram=$(calc_size "$tram")
  253. uram=$(
  254. LANG=C
  255. free | awk '/Mem/ {print $3}'
  256. )
  257. uram=$(calc_size "$uram")
  258. swap=$(
  259. LANG=C
  260. free | awk '/Swap/ {print $2}'
  261. )
  262. swap=$(calc_size "$swap")
  263. uswap=$(
  264. LANG=C
  265. free | awk '/Swap/ {print $3}'
  266. )
  267. uswap=$(calc_size "$uswap")
  268. up=$(awk '{a=$1/86400;b=($1%86400)/3600;c=($1%3600)/60} {printf("%d days, %d hour %d min\n",a,b,c)}' /proc/uptime)
  269. if _exists "w"; then
  270. load=$(
  271. LANG=C
  272. w | head -1 | awk -F'load average:' '{print $2}' | sed 's/^[ \t]*//;s/[ \t]*$//'
  273. )
  274. elif _exists "uptime"; then
  275. load=$(
  276. LANG=C
  277. uptime | head -1 | awk -F'load average:' '{print $2}' | sed 's/^[ \t]*//;s/[ \t]*$//'
  278. )
  279. fi
  280. opsy=$(get_opsy)
  281. arch=$(uname -m)
  282. if _exists "getconf"; then
  283. lbit=$(getconf LONG_BIT)
  284. else
  285. echo "${arch}" | grep -q "64" && lbit="64" || lbit="32"
  286. fi
  287. kern=$(uname -r)
  288. in_kernel_no_swap_total_size=$(
  289. LANG=C
  290. df -t simfs -t ext2 -t ext3 -t ext4 -t btrfs -t xfs -t vfat -t ntfs --total 2>/dev/null | grep total | awk '{ print $2 }'
  291. )
  292. swap_total_size=$(free -k | grep Swap | awk '{print $2}')
  293. zfs_total_size=$(to_kibyte "$(calc_sum "$(zpool list -o size -Hp 2> /dev/null)")")
  294. disk_total_size=$(calc_size $((swap_total_size + in_kernel_no_swap_total_size + zfs_total_size)))
  295. in_kernel_no_swap_used_size=$(
  296. LANG=C
  297. df -t simfs -t ext2 -t ext3 -t ext4 -t btrfs -t xfs -t vfat -t ntfs --total 2>/dev/null | grep total | awk '{ print $3 }'
  298. )
  299. swap_used_size=$(free -k | grep Swap | awk '{print $3}')
  300. zfs_used_size=$(to_kibyte "$(calc_sum "$(zpool list -o allocated -Hp 2> /dev/null)")")
  301. disk_used_size=$(calc_size $((swap_used_size + in_kernel_no_swap_used_size + zfs_used_size)))
  302. tcpctrl=$(sysctl net.ipv4.tcp_congestion_control | awk -F ' ' '{print $3}')
  303. }
  304. # Print System information
  305. print_system_info() {
  306. if [ -n "$cname" ]; then
  307. echo " CPU Model : $(_blue "$cname")"
  308. else
  309. echo " CPU Model : $(_blue "CPU model not detected")"
  310. fi
  311. if [ -n "$freq" ]; then
  312. echo " CPU Cores : $(_blue "$cores @ $freq MHz")"
  313. else
  314. echo " CPU Cores : $(_blue "$cores")"
  315. fi
  316. if [ -n "$ccache" ]; then
  317. echo " CPU Cache : $(_blue "$ccache")"
  318. fi
  319. if [ -n "$cpu_aes" ]; then
  320. echo " AES-NI : $(_green "\xe2\x9c\x93 Enabled")"
  321. else
  322. echo " AES-NI : $(_red "\xe2\x9c\x97 Disabled")"
  323. fi
  324. if [ -n "$cpu_virt" ]; then
  325. echo " VM-x/AMD-V : $(_green "\xe2\x9c\x93 Enabled")"
  326. else
  327. echo " VM-x/AMD-V : $(_red "\xe2\x9c\x97 Disabled")"
  328. fi
  329. echo " Total Disk : $(_yellow "$disk_total_size") $(_blue "($disk_used_size Used)")"
  330. echo " Total Mem : $(_yellow "$tram") $(_blue "($uram Used)")"
  331. if [ "$swap" != "0" ]; then
  332. echo " Total Swap : $(_blue "$swap ($uswap Used)")"
  333. fi
  334. echo " System uptime : $(_blue "$up")"
  335. echo " Load average : $(_blue "$load")"
  336. echo " OS : $(_blue "$opsy")"
  337. echo " Arch : $(_blue "$arch ($lbit Bit)")"
  338. echo " Kernel : $(_blue "$kern")"
  339. echo " TCP CC : $(_yellow "$tcpctrl")"
  340. echo " Virtualization : $(_blue "$virt")"
  341. echo " IPv4/IPv6 : $online"
  342. }
  343. print_io_test() {
  344. freespace=$(df -m . | awk 'NR==2 {print $4}')
  345. if [ -z "${freespace}" ]; then
  346. freespace=$(df -m . | awk 'NR==3 {print $3}')
  347. fi
  348. if [ "${freespace}" -gt 1024 ]; then
  349. writemb=2048
  350. io1=$(io_test ${writemb})
  351. echo " I/O Speed(1st run) : $(_yellow "$io1")"
  352. io2=$(io_test ${writemb})
  353. echo " I/O Speed(2nd run) : $(_yellow "$io2")"
  354. io3=$(io_test ${writemb})
  355. echo " I/O Speed(3rd run) : $(_yellow "$io3")"
  356. ioraw1=$(echo "$io1" | awk 'NR==1 {print $1}')
  357. [[ "$(echo "$io1" | awk 'NR==1 {print $2}')" == "GB/s" ]] && ioraw1=$(awk 'BEGIN{print '"$ioraw1"' * 1024}')
  358. ioraw2=$(echo "$io2" | awk 'NR==1 {print $1}')
  359. [[ "$(echo "$io2" | awk 'NR==1 {print $2}')" == "GB/s" ]] && ioraw2=$(awk 'BEGIN{print '"$ioraw2"' * 1024}')
  360. ioraw3=$(echo "$io3" | awk 'NR==1 {print $1}')
  361. [[ "$(echo "$io3" | awk 'NR==1 {print $2}')" == "GB/s" ]] && ioraw3=$(awk 'BEGIN{print '"$ioraw3"' * 1024}')
  362. ioall=$(awk 'BEGIN{print '"$ioraw1"' + '"$ioraw2"' + '"$ioraw3"'}')
  363. ioavg=$(awk 'BEGIN{printf "%.1f", '"$ioall"' / 3}')
  364. echo " I/O Speed(average) : $(_yellow "$ioavg MB/s")"
  365. else
  366. echo " $(_red "Not enough space for I/O Speed test!")"
  367. fi
  368. }
  369. print_end_time() {
  370. end_time=$(date +%s)
  371. time=$((end_time - start_time))
  372. if [ ${time} -gt 60 ]; then
  373. min=$((time / 60))
  374. sec=$((time % 60))
  375. echo " Finished in : ${min} min ${sec} sec"
  376. else
  377. echo " Finished in : ${time} sec"
  378. fi
  379. date_time=$(date '+%Y-%m-%d %H:%M:%S %Z')
  380. echo " Timestamp : $date_time"
  381. }
  382. ! _exists "wget" && _red "Error: wget command not found.\n" && exit 1
  383. ! _exists "free" && _red "Error: free command not found.\n" && exit 1
  384. # check for curl/wget
  385. _exists "curl" && local_curl=true
  386. # test if the host has IPv4/IPv6 connectivity
  387. [[ -n ${local_curl} ]] && ip_check_cmd="curl -s -m 4" || ip_check_cmd="wget -qO- -T 4"
  388. ipv4_check=$( (ping -4 -c 1 -W 4 ipv4.google.com >/dev/null 2>&1 && echo true) || ${ip_check_cmd} -4 icanhazip.com 2> /dev/null)
  389. ipv6_check=$( (ping -6 -c 1 -W 4 ipv6.google.com >/dev/null 2>&1 && echo true) || ${ip_check_cmd} -6 icanhazip.com 2> /dev/null)
  390. if [[ -z "$ipv4_check" && -z "$ipv6_check" ]]; then
  391. _yellow "Warning: Both IPv4 and IPv6 connectivity were not detected.\n"
  392. fi
  393. [[ -z "$ipv4_check" ]] && online="$(_red "\xe2\x9c\x97 Offline")" || online="$(_green "\xe2\x9c\x93 Online")"
  394. [[ -z "$ipv6_check" ]] && online+=" / $(_red "\xe2\x9c\x97 Offline")" || online+=" / $(_green "\xe2\x9c\x93 Online")"
  395. start_time=$(date +%s)
  396. get_system_info
  397. check_virt
  398. clear
  399. print_intro
  400. next
  401. print_system_info
  402. ipv4_info
  403. next
  404. print_io_test
  405. next
  406. install_speedtest && speed && rm -fr speedtest-cli
  407. next
  408. print_end_time
  409. next