- #!/bin/sh
- #
- # Copyright (C) 2010 segal.di.ubi.pt
- # Copyright (C) 2020 nbembedded.com
- #
- # This is free software, licensed under the GNU General Public License v2.
- #
-
- get_ping_size() {
- ps=$1
- case "$ps" in
- small)
- ps="1"
- ;;
- windows)
- ps="32"
- ;;
- standard)
- ps="56"
- ;;
- big)
- ps="248"
- ;;
- huge)
- ps="1492"
- ;;
- jumbo)
- ps="9000"
- ;;
- *)
- echo "Error: invalid ping_size. ping_size should be either: small, windows, standard, big, huge or jumbo" 1>&2
- echo "Corresponding ping packet sizes (bytes): small=1, windows=32, standard=56, big=248, huge=1492, jumbo=9000" 1>&2
- ;;
- esac
- echo $ps
- }
-
- get_ping_family_flag() {
- family=$1
- case "$family" in
- any)
- family=""
- ;;
- ipv4)
- family="-4"
- ;;
- ipv6)
- family="-6"
- ;;
- *)
- echo "Error: invalid address_family \"$family\". address_family should be one of: any, ipv4, ipv6" 1>&2
- ;;
- esac
- echo $family
- }
-
- reboot_now() {
- reboot &
-
- [ "$1" -ge 1 ] && {
- sleep "$1"
- echo 1 > /proc/sys/kernel/sysrq
- echo b > /proc/sysrq-trigger # Will immediately reboot the system without syncing or unmounting your disks.
- }
- }
-
- watchcat_periodic() {
- failure_period="$1"
- force_reboot_delay="$2"
-
- sleep "$failure_period" && reboot_now "$force_reboot_delay"
- }
-
- watchcat_restart_modemmanager_iface() {
- [ "$2" -gt 0 ] && {
- logger -p daemon.info -t "watchcat[$$]" "Resetting current-bands to 'any' on modem: \"$1\" now."
- /usr/bin/mmcli -m any --set-current-bands=any
- }
- logger -p daemon.info -t "watchcat[$$]" "Reconnecting modem: \"$1\" now."
- /etc/init.d/modemmanager restart
- ifup "$1"
- }
-
- watchcat_restart_network_iface() {
- logger -p daemon.info -t "watchcat[$$]" "Restarting network interface: \"$1\"."
- ip link set "$1" down
- ip link set "$1" up
- }
-
- watchcat_run_script() {
- logger -p daemon.info -t "watchcat[$$]" "Running script \"$1\" for network interface: \"$2\"."
- "$1" "$2"
- }
-
- watchcat_restart_all_network() {
- logger -p daemon.info -t "watchcat[$$]" "Restarting networking now by running: /etc/init.d/network restart"
- /etc/init.d/network restart
- }
-
- watchcat_monitor_network() {
- failure_period="$1"
- ping_hosts="$2"
- ping_frequency_interval="$3"
- ping_size="$4"
- iface="$5"
- mm_iface_name="$6"
- mm_iface_unlock_bands="$7"
- address_family="$8"
- script="$9"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
-
- [ "$time_now" -lt "$failure_period" ] && sleep "$((failure_period - time_now))"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_lastcheck="$time_now"
- time_lastcheck_withinternet="$time_now"
-
- ping_size="$(get_ping_size "$ping_size")"
-
- ping_family="$(get_ping_family_flag "$address_family")"
-
- while true; do
- # account for the time ping took to return. With a ping time of 5s, ping might take more than that, so it is important to avoid even more delay.
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_diff="$((time_now - time_lastcheck))"
-
- [ "$time_diff" -lt "$ping_frequency_interval" ] && sleep "$((ping_frequency_interval - time_diff))"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_lastcheck="$time_now"
-
- for host in $ping_hosts; do
- if [ "$iface" != "" ]; then
- ping_result="$(
- ping "$ping_family" -I "$iface" -s "$ping_size" -c 1 "$host" &> /dev/null
- echo $?
- )"
- else
- ping_result="$(
- ping "$ping_family" -s "$ping_size" -c 1 "$host" &> /dev/null
- echo $?
- )"
- fi
-
- if [ "$ping_result" -eq 0 ]; then
- time_lastcheck_withinternet="$time_now"
- else
- if [ "$script" != "" ]; then
- logger -p daemon.info -t "watchcat[$$]" "Could not reach $host via \"$iface\" for \"$((time_now - time_lastcheck_withinternet))\" seconds. Running script after reaching \"$failure_period\" seconds"
- elif [ "$iface" != "" ]; then
- logger -p daemon.info -t "watchcat[$$]" "Could not reach $host via \"$iface\" for \"$((time_now - time_lastcheck_withinternet))\" seconds. Restarting \"$iface\" after reaching \"$failure_period\" seconds"
- else
- logger -p daemon.info -t "watchcat[$$]" "Could not reach $host for \"$((time_now - time_lastcheck_withinternet))\" seconds. Restarting networking after reaching \"$failure_period\" seconds"
- fi
- fi
- done
-
- [ "$((time_now - time_lastcheck_withinternet))" -ge "$failure_period" ] && {
- if [ "$script" != "" ]; then
- watchcat_run_script "$script" "$iface"
- else
- if [ "$mm_iface_name" != "" ]; then
- watchcat_restart_modemmanager_iface "$mm_iface_name" "$mm_iface_unlock_bands"
- fi
- if [ "$iface" != "" ]; then
- watchcat_restart_network_iface "$iface"
- else
- watchcat_restart_all_network
- fi
- fi
- /etc/init.d/watchcat start
- # Restart timer cycle.
- time_lastcheck_withinternet="$time_now"
- }
-
- done
- }
-
- watchcat_ping() {
- failure_period="$1"
- force_reboot_delay="$2"
- ping_hosts="$3"
- ping_frequency_interval="$4"
- ping_size="$5"
- address_family="$6"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
-
- [ "$time_now" -lt "$failure_period" ] && sleep "$((failure_period - time_now))"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_lastcheck="$time_now"
- time_lastcheck_withinternet="$time_now"
-
- ping_size="$(get_ping_size "$ping_size")"
-
- ping_family="$(get_ping_family_flag "$address_family")"
-
- while true; do
- # account for the time ping took to return. With a ping time of 5s, ping might take more than that, so it is important to avoid even more delay.
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_diff="$((time_now - time_lastcheck))"
-
- [ "$time_diff" -lt "$ping_frequency_interval" ] && sleep "$((ping_frequency_interval - time_diff))"
-
- time_now="$(cat /proc/uptime)"
- time_now="${time_now%%.*}"
- time_lastcheck="$time_now"
-
- for host in $ping_hosts; do
- if [ "$iface" != "" ]; then
- ping_result="$(
- ping "$ping_family" -I "$iface" -s "$ping_size" -c 1 "$host" &> /dev/null
- echo $?
- )"
- else
- ping_result="$(
- ping "$ping_family" -s "$ping_size" -c 1 "$host" &> /dev/null
- echo $?
- )"
- fi
-
- if [ "$ping_result" -eq 0 ]; then
- time_lastcheck_withinternet="$time_now"
- else
- logger -p daemon.info -t "watchcat[$$]" "Could not reach $host for $((time_now - time_lastcheck_withinternet)). Rebooting after reaching $failure_period"
- fi
- done
-
- [ "$((time_now - time_lastcheck_withinternet))" -ge "$failure_period" ] && reboot_now "$force_reboot_delay"
- done
- }
-
- mode="$1"
-
- # Fix potential typo in mode and provide backward compatibility.
- [ "$mode" = "allways" ] && mode="periodic_reboot"
- [ "$mode" = "always" ] && mode="periodic_reboot"
- [ "$mode" = "ping" ] && mode="ping_reboot"
-
- case "$mode" in
- periodic_reboot)
- # args from init script: period forcedelay
- watchcat_periodic "$2" "$3"
- ;;
- ping_reboot)
- # args from init script: period forcedelay pinghosts pingperiod pingsize addressfamily
- watchcat_ping "$2" "$3" "$4" "$5" "$6" "$7"
- ;;
- restart_iface)
- # args from init script: period pinghosts pingperiod pingsize interface mmifacename unlockbands addressfamily
- watchcat_monitor_network "$2" "$3" "$4" "$5" "$6" "$7" "$8" "$9" ""
- ;;
- run_script)
- # args from init script: period pinghosts pingperiod pingsize interface addressfamily script
- watchcat_monitor_network "$2" "$3" "$4" "$5" "$6" "" "" "$7" "$8"
- ;;
- *)
- echo "Error: invalid mode selected: $mode"
- ;;
- esac
|